Blame ext/pango/gsttextrender.c

Packit 971217
/* GStreamer
Packit 971217
 * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
Packit 971217
 * Copyright (C) <2003> David Schleef <ds@schleef.org>
Packit 971217
 * Copyright (C) <2009> Young-Ho Cha <ganadist@gmail.com>
Packit 971217
 *
Packit 971217
 * This library is free software; you can redistribute it and/or
Packit 971217
 * modify it under the terms of the GNU Library General Public
Packit 971217
 * License as published by the Free Software Foundation; either
Packit 971217
 * version 2 of the License, or (at your option) any later version.
Packit 971217
 *
Packit 971217
 * This library is distributed in the hope that it will be useful,
Packit 971217
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 971217
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 971217
 * Library General Public License for more details.
Packit 971217
 *
Packit 971217
 * You should have received a copy of the GNU Library General Public
Packit 971217
 * License along with this library; if not, write to the
Packit 971217
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit 971217
 * Boston, MA 02110-1301, USA.
Packit 971217
 */
Packit 971217
Packit 971217
Packit 971217
/**
Packit 971217
 * SECTION:element-textrender
Packit 971217
 * @title: textrender
Packit 971217
 * @see_also: #GstTextOverlay
Packit 971217
 *
Packit 971217
 * This plugin renders text received on the text sink pad to a video
Packit 971217
 * buffer (retaining the alpha channel), so it can later be overlayed
Packit 971217
 * on top of video streams using other elements.
Packit 971217
 *
Packit 971217
 * The text can contain newline characters. (FIXME: What about text
Packit 971217
 * wrapping? It does not make sense in this context)
Packit 971217
 *
Packit 971217
 * ## Example launch lines
Packit 971217
 * |[
Packit 971217
 * gst-launch-1.0 -v filesrc location=subtitles.srt ! subparse ! textrender ! videoconvert ! autovideosink
Packit 971217
 * ]|
Packit 971217
 *
Packit 971217
 */
Packit 971217
Packit 971217
#ifdef HAVE_CONFIG_H
Packit 971217
#include <config.h>
Packit 971217
#endif
Packit 971217
Packit 971217
#include <gst/gst.h>
Packit 971217
#include <gst/video/video.h>
Packit 971217
Packit 971217
#include "gsttextrender.h"
Packit 971217
#include <string.h>
Packit 971217
Packit 971217
#if G_BYTE_ORDER == G_LITTLE_ENDIAN
Packit 971217
# define CAIRO_ARGB_A 3
Packit 971217
# define CAIRO_ARGB_R 2
Packit 971217
# define CAIRO_ARGB_G 1
Packit 971217
# define CAIRO_ARGB_B 0
Packit 971217
#else
Packit 971217
# define CAIRO_ARGB_A 0
Packit 971217
# define CAIRO_ARGB_R 1
Packit 971217
# define CAIRO_ARGB_G 2
Packit 971217
# define CAIRO_ARGB_B 3
Packit 971217
#endif
Packit 971217
Packit 971217
GST_DEBUG_CATEGORY_EXTERN (pango_debug);
Packit 971217
#define GST_CAT_DEFAULT pango_debug
Packit 971217
Packit 971217
#define MINIMUM_OUTLINE_OFFSET 1.0
Packit 971217
Packit 971217
#define DEFAULT_PROP_VALIGNMENT GST_TEXT_RENDER_VALIGN_BASELINE
Packit 971217
#define DEFAULT_PROP_HALIGNMENT GST_TEXT_RENDER_HALIGN_CENTER
Packit 971217
#define DEFAULT_PROP_LINE_ALIGNMENT GST_TEXT_RENDER_LINE_ALIGN_CENTER
Packit 971217
#define DEFAULT_PROP_XPAD       25
Packit 971217
#define DEFAULT_PROP_YPAD       25
Packit 971217
Packit 971217
#define DEFAULT_RENDER_WIDTH 720
Packit 971217
#define DEFAULT_RENDER_HEIGHT 576
Packit 971217
Packit 971217
enum
Packit 971217
{
Packit 971217
  PROP_0,
Packit 971217
  PROP_HALIGNMENT,
Packit 971217
  PROP_VALIGNMENT,
Packit 971217
  PROP_LINE_ALIGNMENT,
Packit 971217
  PROP_XPAD,
Packit 971217
  PROP_YPAD,
Packit 971217
  PROP_FONT_DESC
Packit 971217
};
Packit 971217
Packit 971217
#define VIDEO_FORMATS "{ AYUV, ARGB } "
Packit 971217
Packit 971217
static GstStaticPadTemplate src_template_factory =
Packit 971217
GST_STATIC_PAD_TEMPLATE ("src",
Packit 971217
    GST_PAD_SRC,
Packit 971217
    GST_PAD_ALWAYS,
Packit 971217
    GST_STATIC_CAPS (GST_VIDEO_CAPS_MAKE (VIDEO_FORMATS))
Packit 971217
    );
Packit 971217
Packit 971217
static GstStaticPadTemplate sink_template_factory =
Packit 971217
GST_STATIC_PAD_TEMPLATE ("sink",
Packit 971217
    GST_PAD_SINK,
Packit 971217
    GST_PAD_ALWAYS,
Packit 971217
    GST_STATIC_CAPS ("text/x-raw, format = { pango-markup, utf8 }")
Packit 971217
    );
Packit 971217
Packit 971217
#define GST_TYPE_TEXT_RENDER_VALIGN (gst_text_render_valign_get_type())
Packit 971217
static GType
Packit 971217
gst_text_render_valign_get_type (void)
Packit 971217
{
Packit 971217
  static GType text_render_valign_type = 0;
Packit 971217
  static const GEnumValue text_render_valign[] = {
Packit 971217
    {GST_TEXT_RENDER_VALIGN_BASELINE, "baseline", "baseline"},
Packit 971217
    {GST_TEXT_RENDER_VALIGN_BOTTOM, "bottom", "bottom"},
Packit 971217
    {GST_TEXT_RENDER_VALIGN_TOP, "top", "top"},
Packit 971217
    {0, NULL, NULL},
Packit 971217
  };
Packit 971217
Packit 971217
  if (!text_render_valign_type) {
Packit 971217
    text_render_valign_type =
Packit 971217
        g_enum_register_static ("GstTextRenderVAlign", text_render_valign);
Packit 971217
  }
Packit 971217
  return text_render_valign_type;
Packit 971217
}
Packit 971217
Packit 971217
#define GST_TYPE_TEXT_RENDER_HALIGN (gst_text_render_halign_get_type())
Packit 971217
static GType
Packit 971217
gst_text_render_halign_get_type (void)
Packit 971217
{
Packit 971217
  static GType text_render_halign_type = 0;
Packit 971217
  static const GEnumValue text_render_halign[] = {
Packit 971217
    {GST_TEXT_RENDER_HALIGN_LEFT, "left", "left"},
Packit 971217
    {GST_TEXT_RENDER_HALIGN_CENTER, "center", "center"},
Packit 971217
    {GST_TEXT_RENDER_HALIGN_RIGHT, "right", "right"},
Packit 971217
    {0, NULL, NULL},
Packit 971217
  };
Packit 971217
Packit 971217
  if (!text_render_halign_type) {
Packit 971217
    text_render_halign_type =
Packit 971217
        g_enum_register_static ("GstTextRenderHAlign", text_render_halign);
Packit 971217
  }
Packit 971217
  return text_render_halign_type;
Packit 971217
}
Packit 971217
Packit 971217
#define GST_TYPE_TEXT_RENDER_LINE_ALIGN (gst_text_render_line_align_get_type())
Packit 971217
static GType
Packit 971217
gst_text_render_line_align_get_type (void)
Packit 971217
{
Packit 971217
  static GType text_render_line_align_type = 0;
Packit 971217
  static const GEnumValue text_render_line_align[] = {
Packit 971217
    {GST_TEXT_RENDER_LINE_ALIGN_LEFT, "left", "left"},
Packit 971217
    {GST_TEXT_RENDER_LINE_ALIGN_CENTER, "center", "center"},
Packit 971217
    {GST_TEXT_RENDER_LINE_ALIGN_RIGHT, "right", "right"},
Packit 971217
    {0, NULL, NULL}
Packit 971217
  };
Packit 971217
Packit 971217
  if (!text_render_line_align_type) {
Packit 971217
    text_render_line_align_type =
Packit 971217
        g_enum_register_static ("GstTextRenderLineAlign",
Packit 971217
        text_render_line_align);
Packit 971217
  }
Packit 971217
  return text_render_line_align_type;
Packit 971217
}
Packit 971217
Packit 971217
static void gst_text_render_adjust_values_with_fontdesc (GstTextRender *
Packit 971217
    render, PangoFontDescription * desc);
Packit 971217
Packit 971217
#define gst_text_render_parent_class parent_class
Packit 971217
G_DEFINE_TYPE (GstTextRender, gst_text_render, GST_TYPE_ELEMENT);
Packit 971217
Packit 971217
static void gst_text_render_finalize (GObject * object);
Packit 971217
static void gst_text_render_set_property (GObject * object,
Packit 971217
    guint prop_id, const GValue * value, GParamSpec * pspec);
Packit 971217
static void gst_text_render_get_property (GObject * object,
Packit 971217
    guint prop_id, GValue * value, GParamSpec * pspec);
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_class_init (GstTextRenderClass * klass)
Packit 971217
{
Packit 971217
  GObjectClass *gobject_class;
Packit 971217
  GstElementClass *gstelement_class;
Packit 971217
  PangoFontMap *fontmap;
Packit 971217
Packit 971217
  gobject_class = (GObjectClass *) klass;
Packit 971217
  gstelement_class = (GstElementClass *) klass;
Packit 971217
Packit 971217
  parent_class = g_type_class_peek_parent (klass);
Packit 971217
Packit 971217
  gobject_class->finalize = gst_text_render_finalize;
Packit 971217
  gobject_class->set_property = gst_text_render_set_property;
Packit 971217
  gobject_class->get_property = gst_text_render_get_property;
Packit 971217
Packit 971217
  gst_element_class_add_static_pad_template (gstelement_class,
Packit 971217
      &src_template_factory);
Packit 971217
  gst_element_class_add_static_pad_template (gstelement_class,
Packit 971217
      &sink_template_factory);
Packit 971217
Packit 971217
  gst_element_class_set_static_metadata (gstelement_class, "Text renderer",
Packit 971217
      "Filter/Editor/Video",
Packit 971217
      "Renders a text string to an image bitmap",
Packit 971217
      "David Schleef <ds@schleef.org>, "
Packit 971217
      "GStreamer maintainers <gstreamer-devel@lists.freedesktop.org>");
Packit 971217
Packit 971217
  fontmap = pango_cairo_font_map_get_default ();
Packit 971217
  klass->pango_context =
Packit 971217
      pango_font_map_create_context (PANGO_FONT_MAP (fontmap));
Packit 971217
  g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_FONT_DESC,
Packit 971217
      g_param_spec_string ("font-desc", "font description",
Packit 971217
          "Pango font description of font "
Packit 971217
          "to be used for rendering. "
Packit 971217
          "See documentation of "
Packit 971217
          "pango_font_description_from_string"
Packit 971217
          " for syntax.", "", G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS));
Packit 971217
  g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_VALIGNMENT,
Packit 971217
      g_param_spec_enum ("valignment", "vertical alignment",
Packit 971217
          "Vertical alignment of the text", GST_TYPE_TEXT_RENDER_VALIGN,
Packit 971217
          DEFAULT_PROP_VALIGNMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit 971217
  g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_HALIGNMENT,
Packit 971217
      g_param_spec_enum ("halignment", "horizontal alignment",
Packit 971217
          "Horizontal alignment of the text", GST_TYPE_TEXT_RENDER_HALIGN,
Packit 971217
          DEFAULT_PROP_HALIGNMENT, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit 971217
  g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_XPAD,
Packit 971217
      g_param_spec_int ("xpad", "horizontal paddding",
Packit 971217
          "Horizontal paddding when using left/right alignment", 0, G_MAXINT,
Packit 971217
          DEFAULT_PROP_XPAD, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit 971217
  g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_YPAD,
Packit 971217
      g_param_spec_int ("ypad", "vertical padding",
Packit 971217
          "Vertical padding when using top/bottom alignment", 0, G_MAXINT,
Packit 971217
          DEFAULT_PROP_YPAD, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit 971217
  g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_LINE_ALIGNMENT,
Packit 971217
      g_param_spec_enum ("line-alignment", "line alignment",
Packit 971217
          "Alignment of text lines relative to each other.",
Packit 971217
          GST_TYPE_TEXT_RENDER_LINE_ALIGN, DEFAULT_PROP_LINE_ALIGNMENT,
Packit 971217
          G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_adjust_values_with_fontdesc (GstTextRender * render,
Packit 971217
    PangoFontDescription * desc)
Packit 971217
{
Packit 971217
  gint font_size = pango_font_description_get_size (desc) / PANGO_SCALE;
Packit 971217
Packit 971217
  render->shadow_offset = (double) (font_size) / 13.0;
Packit 971217
  render->outline_offset = (double) (font_size) / 15.0;
Packit 971217
  if (render->outline_offset < MINIMUM_OUTLINE_OFFSET)
Packit 971217
    render->outline_offset = MINIMUM_OUTLINE_OFFSET;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_render_pangocairo (GstTextRender * render)
Packit 971217
{
Packit 971217
  cairo_t *cr;
Packit 971217
  cairo_surface_t *surface;
Packit 971217
  cairo_t *cr_shadow;
Packit 971217
  cairo_surface_t *surface_shadow;
Packit 971217
  PangoRectangle ink_rect, logical_rect;
Packit 971217
  gint width, height;
Packit 971217
Packit 971217
  pango_layout_get_pixel_extents (render->layout, &ink_rect, &logical_rect);
Packit 971217
Packit 971217
  width = logical_rect.width + render->shadow_offset;
Packit 971217
  height = logical_rect.height + logical_rect.y + render->shadow_offset;
Packit 971217
Packit 971217
  surface_shadow = cairo_image_surface_create (CAIRO_FORMAT_A8, width, height);
Packit 971217
  cr_shadow = cairo_create (surface_shadow);
Packit 971217
Packit 971217
  /* clear shadow surface */
Packit 971217
  cairo_set_operator (cr_shadow, CAIRO_OPERATOR_CLEAR);
Packit 971217
  cairo_paint (cr_shadow);
Packit 971217
  cairo_set_operator (cr_shadow, CAIRO_OPERATOR_OVER);
Packit 971217
Packit 971217
  /* draw shadow text */
Packit 971217
  cairo_save (cr_shadow);
Packit 971217
  cairo_set_source_rgba (cr_shadow, 0.0, 0.0, 0.0, 0.5);
Packit 971217
  cairo_translate (cr_shadow, render->shadow_offset, render->shadow_offset);
Packit 971217
  pango_cairo_show_layout (cr_shadow, render->layout);
Packit 971217
  cairo_restore (cr_shadow);
Packit 971217
Packit 971217
  /* draw outline text */
Packit 971217
  cairo_save (cr_shadow);
Packit 971217
  cairo_set_source_rgb (cr_shadow, 0.0, 0.0, 0.0);
Packit 971217
  cairo_set_line_width (cr_shadow, render->outline_offset);
Packit 971217
  pango_cairo_layout_path (cr_shadow, render->layout);
Packit 971217
  cairo_stroke (cr_shadow);
Packit 971217
  cairo_restore (cr_shadow);
Packit 971217
Packit 971217
  cairo_destroy (cr_shadow);
Packit 971217
Packit 971217
  render->text_image = g_realloc (render->text_image, 4 * width * height);
Packit 971217
Packit 971217
  surface = cairo_image_surface_create_for_data (render->text_image,
Packit 971217
      CAIRO_FORMAT_ARGB32, width, height, width * 4);
Packit 971217
  cr = cairo_create (surface);
Packit 971217
  cairo_set_operator (cr, CAIRO_OPERATOR_CLEAR);
Packit 971217
  cairo_paint (cr);
Packit 971217
  cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
Packit 971217
Packit 971217
  /* set default color */
Packit 971217
  cairo_set_source_rgb (cr, 1.0, 1.0, 1.0);
Packit 971217
Packit 971217
  cairo_save (cr);
Packit 971217
  /* draw text */
Packit 971217
  pango_cairo_show_layout (cr, render->layout);
Packit 971217
  cairo_restore (cr);
Packit 971217
Packit 971217
  /* composite shadow with offset */
Packit 971217
  cairo_set_operator (cr, CAIRO_OPERATOR_DEST_OVER);
Packit 971217
  cairo_set_source_surface (cr, surface_shadow, 0.0, 0.0);
Packit 971217
  cairo_paint (cr);
Packit 971217
Packit 971217
  cairo_destroy (cr);
Packit 971217
  cairo_surface_destroy (surface_shadow);
Packit 971217
  cairo_surface_destroy (surface);
Packit 971217
  render->image_width = width;
Packit 971217
  render->image_height = height;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_check_argb (GstTextRender * render)
Packit 971217
{
Packit 971217
  GstCaps *peer_caps;
Packit 971217
  peer_caps = gst_pad_get_allowed_caps (render->srcpad);
Packit 971217
  if (G_LIKELY (peer_caps)) {
Packit 971217
    guint i = 0, n = 0;
Packit 971217
Packit 971217
    n = gst_caps_get_size (peer_caps);
Packit 971217
    GST_DEBUG_OBJECT (render, "peer allowed caps (%u structure(s)) are %"
Packit 971217
        GST_PTR_FORMAT, n, peer_caps);
Packit 971217
Packit 971217
    /* Check if AYUV or ARGB is first */
Packit 971217
    for (i = 0; i < n; i++) {
Packit 971217
      GstStructure *s;
Packit 971217
      GstVideoFormat vformat;
Packit 971217
      const GstVideoFormatInfo *info;
Packit 971217
      const gchar *fmt;
Packit 971217
Packit 971217
      s = gst_caps_get_structure (peer_caps, i);
Packit 971217
      if (!gst_structure_has_name (s, "video/x-raw"))
Packit 971217
        continue;
Packit 971217
Packit 971217
      fmt = gst_structure_get_string (s, "format");
Packit 971217
      if (fmt == NULL)
Packit 971217
        continue;
Packit 971217
Packit 971217
      vformat = gst_video_format_from_string (fmt);
Packit 971217
      info = gst_video_format_get_info (vformat);
Packit 971217
      if (info == NULL)
Packit 971217
        continue;
Packit 971217
Packit 971217
      render->use_ARGB = GST_VIDEO_FORMAT_INFO_HAS_ALPHA (info);
Packit 971217
    }
Packit 971217
    gst_caps_unref (peer_caps);
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static gboolean
Packit 971217
gst_text_render_src_setcaps (GstTextRender * render, GstCaps * caps)
Packit 971217
{
Packit 971217
  GstStructure *structure;
Packit 971217
  gboolean ret;
Packit 971217
  gint width = 0, height = 0;
Packit 971217
Packit 971217
  structure = gst_caps_get_structure (caps, 0);
Packit 971217
  gst_structure_get_int (structure, "width", &width);
Packit 971217
  gst_structure_get_int (structure, "height", &height);
Packit 971217
Packit 971217
  GST_DEBUG_OBJECT (render, "Got caps %" GST_PTR_FORMAT, caps);
Packit 971217
Packit 971217
  if (width >= render->image_width && height >= render->image_height) {
Packit 971217
    render->width = width;
Packit 971217
    render->height = height;
Packit 971217
  }
Packit 971217
Packit 971217
  gst_text_render_check_argb (render);
Packit 971217
Packit 971217
  ret = gst_pad_set_caps (render->srcpad, caps);
Packit 971217
Packit 971217
  return ret;
Packit 971217
}
Packit 971217
Packit 971217
static GstCaps *
Packit 971217
gst_text_render_fixate_caps (GstTextRender * render, GstCaps * caps)
Packit 971217
{
Packit 971217
  GstStructure *s;
Packit 971217
Packit 971217
  caps = gst_caps_truncate (caps);
Packit 971217
Packit 971217
  caps = gst_caps_make_writable (caps);
Packit 971217
  s = gst_caps_get_structure (caps, 0);
Packit 971217
Packit 971217
  GST_DEBUG ("Fixating caps %" GST_PTR_FORMAT, caps);
Packit 971217
  gst_structure_fixate_field_nearest_int (s, "width", MAX (render->image_width,
Packit 971217
          DEFAULT_RENDER_WIDTH));
Packit 971217
  gst_structure_fixate_field_nearest_int (s, "height",
Packit 971217
      MAX (render->image_height + render->ypad, DEFAULT_RENDER_HEIGHT));
Packit 971217
  caps = gst_caps_fixate (caps);
Packit 971217
  GST_DEBUG ("Fixated to    %" GST_PTR_FORMAT, caps);
Packit 971217
Packit 971217
  return caps;
Packit 971217
}
Packit 971217
Packit 971217
#define CAIRO_UNPREMULTIPLY(a,r,g,b) G_STMT_START { \
Packit 971217
  b = (a > 0) ? MIN ((b * 255 + a / 2) / a, 255) : 0; \
Packit 971217
  g = (a > 0) ? MIN ((g * 255 + a / 2) / a, 255) : 0; \
Packit 971217
  r = (a > 0) ? MIN ((r * 255 + a / 2) / a, 255) : 0; \
Packit 971217
} G_STMT_END
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_renderer_image_to_ayuv (GstTextRender * render, guchar * pixbuf,
Packit 971217
    int xpos, int ypos, int stride)
Packit 971217
{
Packit 971217
  int y;                        /* text bitmap coordinates */
Packit 971217
  guchar *p, *bitp;
Packit 971217
  guchar a, r, g, b;
Packit 971217
  int width, height;
Packit 971217
Packit 971217
  width = render->image_width;
Packit 971217
  height = render->image_height;
Packit 971217
Packit 971217
  for (y = 0; y < height && ypos + y < render->height; y++) {
Packit 971217
    int n;
Packit 971217
    p = pixbuf + (ypos + y) * stride + xpos * 4;
Packit 971217
    bitp = render->text_image + y * width * 4;
Packit 971217
    for (n = 0; n < width && n < render->width; n++) {
Packit 971217
      b = bitp[CAIRO_ARGB_B];
Packit 971217
      g = bitp[CAIRO_ARGB_G];
Packit 971217
      r = bitp[CAIRO_ARGB_R];
Packit 971217
      a = bitp[CAIRO_ARGB_A];
Packit 971217
      bitp += 4;
Packit 971217
Packit 971217
      /* Cairo uses pre-multiplied ARGB, unpremultiply it */
Packit 971217
      CAIRO_UNPREMULTIPLY (a, r, g, b);
Packit 971217
Packit 971217
      *p++ = a;
Packit 971217
      *p++ = CLAMP ((int) (((19595 * r) >> 16) + ((38470 * g) >> 16) +
Packit 971217
              ((7471 * b) >> 16)), 0, 255);
Packit 971217
      *p++ = CLAMP ((int) (-((11059 * r) >> 16) - ((21709 * g) >> 16) +
Packit 971217
              ((32768 * b) >> 16) + 128), 0, 255);
Packit 971217
      *p++ = CLAMP ((int) (((32768 * r) >> 16) - ((27439 * g) >> 16) -
Packit 971217
              ((5329 * b) >> 16) + 128), 0, 255);
Packit 971217
    }
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_renderer_image_to_argb (GstTextRender * render, guchar * pixbuf,
Packit 971217
    int xpos, int ypos, int stride)
Packit 971217
{
Packit 971217
  int i, j;
Packit 971217
  guchar *p, *bitp;
Packit 971217
  int width, height;
Packit 971217
Packit 971217
  width = render->image_width;
Packit 971217
  height = render->image_height;
Packit 971217
Packit 971217
  for (i = 0; i < height && ypos + i < render->height; i++) {
Packit 971217
    p = pixbuf + (ypos + i) * stride + xpos * 4;
Packit 971217
    bitp = render->text_image + i * width * 4;
Packit 971217
    for (j = 0; j < width && j < render->width; j++) {
Packit 971217
      p[0] = bitp[CAIRO_ARGB_A];
Packit 971217
      p[1] = bitp[CAIRO_ARGB_R];
Packit 971217
      p[2] = bitp[CAIRO_ARGB_G];
Packit 971217
      p[3] = bitp[CAIRO_ARGB_B];
Packit 971217
Packit 971217
      /* Cairo uses pre-multiplied ARGB, unpremultiply it */
Packit 971217
      CAIRO_UNPREMULTIPLY (p[0], p[1], p[2], p[3]);
Packit 971217
Packit 971217
      bitp += 4;
Packit 971217
      p += 4;
Packit 971217
    }
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static GstFlowReturn
Packit 971217
gst_text_render_chain (GstPad * pad, GstObject * parent, GstBuffer * inbuf)
Packit 971217
{
Packit 971217
  GstTextRender *render;
Packit 971217
  GstFlowReturn ret;
Packit 971217
  GstBuffer *outbuf;
Packit 971217
  GstCaps *caps = NULL, *padcaps;
Packit 971217
  GstMapInfo map;
Packit 971217
  guint8 *data;
Packit 971217
  gsize size;
Packit 971217
  gint n;
Packit 971217
  gint xpos, ypos;
Packit 971217
Packit 971217
  render = GST_TEXT_RENDER (parent);
Packit 971217
Packit 971217
  gst_buffer_map (inbuf, &map, GST_MAP_READ);
Packit 971217
  data = map.data;
Packit 971217
  size = map.size;
Packit 971217
Packit 971217
  /* somehow pango barfs over "\0" buffers... */
Packit 971217
  while (size > 0 &&
Packit 971217
      (data[size - 1] == '\r' ||
Packit 971217
          data[size - 1] == '\n' || data[size - 1] == '\0')) {
Packit 971217
    size--;
Packit 971217
  }
Packit 971217
Packit 971217
  /* render text */
Packit 971217
  GST_DEBUG ("rendering '%*s'", (gint) size, data);
Packit 971217
  pango_layout_set_markup (render->layout, (gchar *) data, size);
Packit 971217
  gst_text_render_render_pangocairo (render);
Packit 971217
  gst_buffer_unmap (inbuf, &map);
Packit 971217
Packit 971217
  gst_text_render_check_argb (render);
Packit 971217
Packit 971217
  padcaps = gst_pad_query_caps (render->srcpad, NULL);
Packit 971217
  caps = gst_pad_peer_query_caps (render->srcpad, padcaps);
Packit 971217
  gst_caps_unref (padcaps);
Packit 971217
Packit 971217
  if (!caps || gst_caps_is_empty (caps)) {
Packit 971217
    GST_ELEMENT_ERROR (render, CORE, NEGOTIATION, (NULL), (NULL));
Packit 971217
    ret = GST_FLOW_ERROR;
Packit 971217
    goto done;
Packit 971217
  }
Packit 971217
Packit 971217
  caps = gst_text_render_fixate_caps (render, caps);
Packit 971217
Packit 971217
  if (!gst_text_render_src_setcaps (render, caps)) {
Packit 971217
    GST_ELEMENT_ERROR (render, CORE, NEGOTIATION, (NULL), (NULL));
Packit 971217
    ret = GST_FLOW_ERROR;
Packit 971217
    goto done;
Packit 971217
  }
Packit 971217
Packit 971217
  if (render->segment_event) {
Packit 971217
    gst_pad_push_event (render->srcpad, render->segment_event);
Packit 971217
    render->segment_event = NULL;
Packit 971217
  }
Packit 971217
Packit 971217
  GST_DEBUG ("Allocating buffer WxH = %dx%d", render->width, render->height);
Packit 971217
  outbuf = gst_buffer_new_and_alloc (render->width * render->height * 4);
Packit 971217
Packit 971217
  gst_buffer_copy_into (outbuf, inbuf, GST_BUFFER_COPY_TIMESTAMPS, 0, -1);
Packit 971217
Packit 971217
  gst_buffer_map (outbuf, &map, GST_MAP_WRITE);
Packit 971217
  data = map.data;
Packit 971217
  size = map.size;
Packit 971217
Packit 971217
  if (render->use_ARGB) {
Packit 971217
    memset (data, 0, render->width * render->height * 4);
Packit 971217
  } else {
Packit 971217
    for (n = 0; n < render->width * render->height; n++) {
Packit 971217
      data[n * 4] = data[n * 4 + 1] = 0;
Packit 971217
      data[n * 4 + 2] = data[n * 4 + 3] = 128;
Packit 971217
    }
Packit 971217
  }
Packit 971217
Packit 971217
  switch (render->halign) {
Packit 971217
    case GST_TEXT_RENDER_HALIGN_LEFT:
Packit 971217
      xpos = render->xpad;
Packit 971217
      break;
Packit 971217
    case GST_TEXT_RENDER_HALIGN_CENTER:
Packit 971217
      xpos = (render->width - render->image_width) / 2;
Packit 971217
      break;
Packit 971217
    case GST_TEXT_RENDER_HALIGN_RIGHT:
Packit 971217
      xpos = render->width - render->image_width - render->xpad;
Packit 971217
      break;
Packit 971217
    default:
Packit 971217
      xpos = 0;
Packit 971217
  }
Packit 971217
Packit 971217
  switch (render->valign) {
Packit 971217
    case GST_TEXT_RENDER_VALIGN_BOTTOM:
Packit 971217
      ypos = render->height - render->image_height - render->ypad;
Packit 971217
      break;
Packit 971217
    case GST_TEXT_RENDER_VALIGN_BASELINE:
Packit 971217
      ypos = render->height - (render->image_height + render->ypad);
Packit 971217
      break;
Packit 971217
    case GST_TEXT_RENDER_VALIGN_TOP:
Packit 971217
      ypos = render->ypad;
Packit 971217
      break;
Packit 971217
    default:
Packit 971217
      ypos = render->ypad;
Packit 971217
      break;
Packit 971217
  }
Packit 971217
Packit 971217
  if (render->text_image) {
Packit 971217
    if (render->use_ARGB) {
Packit 971217
      gst_text_renderer_image_to_argb (render, data, xpos, ypos,
Packit 971217
          render->width * 4);
Packit 971217
    } else {
Packit 971217
      gst_text_renderer_image_to_ayuv (render, data, xpos, ypos,
Packit 971217
          render->width * 4);
Packit 971217
    }
Packit 971217
  }
Packit 971217
  gst_buffer_unmap (outbuf, &map);
Packit 971217
Packit 971217
  ret = gst_pad_push (render->srcpad, outbuf);
Packit 971217
Packit 971217
done:
Packit 971217
  if (caps)
Packit 971217
    gst_caps_unref (caps);
Packit 971217
  gst_buffer_unref (inbuf);
Packit 971217
Packit 971217
  return ret;
Packit 971217
}
Packit 971217
Packit 971217
static gboolean
Packit 971217
gst_text_render_event (GstPad * pad, GstObject * parent, GstEvent * event)
Packit 971217
{
Packit 971217
  GstTextRender *render = GST_TEXT_RENDER (parent);
Packit 971217
  gboolean ret = TRUE;
Packit 971217
Packit 971217
  switch (GST_EVENT_TYPE (event)) {
Packit 971217
    case GST_EVENT_SEGMENT:
Packit 971217
    {
Packit 971217
      if (gst_pad_has_current_caps (render->srcpad)) {
Packit 971217
        ret = gst_pad_push_event (render->srcpad, event);
Packit 971217
      } else {
Packit 971217
        gst_event_replace (&render->segment_event, event);
Packit 971217
        gst_event_unref (event);
Packit 971217
      }
Packit 971217
      break;
Packit 971217
    }
Packit 971217
    default:
Packit 971217
      ret = gst_pad_push_event (render->srcpad, event);
Packit 971217
      break;
Packit 971217
  }
Packit 971217
Packit 971217
  return ret;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_finalize (GObject * object)
Packit 971217
{
Packit 971217
  GstTextRender *render = GST_TEXT_RENDER (object);
Packit 971217
Packit 971217
  gst_event_replace (&render->segment_event, NULL);
Packit 971217
Packit 971217
  g_free (render->text_image);
Packit 971217
Packit 971217
  if (render->layout)
Packit 971217
    g_object_unref (render->layout);
Packit 971217
Packit 971217
  G_OBJECT_CLASS (parent_class)->finalize (object);
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_init (GstTextRender * render)
Packit 971217
{
Packit 971217
  GstPadTemplate *template;
Packit 971217
Packit 971217
  /* sink */
Packit 971217
  template = gst_static_pad_template_get (&sink_template_factory);
Packit 971217
  render->sinkpad = gst_pad_new_from_template (template, "sink");
Packit 971217
  gst_object_unref (template);
Packit 971217
  gst_pad_set_chain_function (render->sinkpad,
Packit 971217
      GST_DEBUG_FUNCPTR (gst_text_render_chain));
Packit 971217
  gst_pad_set_event_function (render->sinkpad,
Packit 971217
      GST_DEBUG_FUNCPTR (gst_text_render_event));
Packit 971217
Packit 971217
  gst_element_add_pad (GST_ELEMENT (render), render->sinkpad);
Packit 971217
Packit 971217
  /* source */
Packit 971217
  template = gst_static_pad_template_get (&src_template_factory);
Packit 971217
  render->srcpad = gst_pad_new_from_template (template, "src");
Packit 971217
  gst_object_unref (template);
Packit 971217
Packit 971217
  gst_element_add_pad (GST_ELEMENT (render), render->srcpad);
Packit 971217
Packit 971217
  render->line_align = DEFAULT_PROP_LINE_ALIGNMENT;
Packit 971217
  render->layout =
Packit 971217
      pango_layout_new (GST_TEXT_RENDER_GET_CLASS (render)->pango_context);
Packit 971217
  pango_layout_set_alignment (render->layout,
Packit 971217
      (PangoAlignment) render->line_align);
Packit 971217
Packit 971217
  render->halign = DEFAULT_PROP_HALIGNMENT;
Packit 971217
  render->valign = DEFAULT_PROP_VALIGNMENT;
Packit 971217
  render->xpad = DEFAULT_PROP_XPAD;
Packit 971217
  render->ypad = DEFAULT_PROP_YPAD;
Packit 971217
Packit 971217
  render->width = DEFAULT_RENDER_WIDTH;
Packit 971217
  render->height = DEFAULT_RENDER_HEIGHT;
Packit 971217
Packit 971217
  render->use_ARGB = FALSE;
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_set_property (GObject * object, guint prop_id,
Packit 971217
    const GValue * value, GParamSpec * pspec)
Packit 971217
{
Packit 971217
  GstTextRender *render = GST_TEXT_RENDER (object);
Packit 971217
Packit 971217
  switch (prop_id) {
Packit 971217
    case PROP_VALIGNMENT:
Packit 971217
      render->valign = g_value_get_enum (value);
Packit 971217
      break;
Packit 971217
    case PROP_HALIGNMENT:
Packit 971217
      render->halign = g_value_get_enum (value);
Packit 971217
      break;
Packit 971217
    case PROP_LINE_ALIGNMENT:
Packit 971217
      render->line_align = g_value_get_enum (value);
Packit 971217
      pango_layout_set_alignment (render->layout,
Packit 971217
          (PangoAlignment) render->line_align);
Packit 971217
      break;
Packit 971217
    case PROP_XPAD:
Packit 971217
      render->xpad = g_value_get_int (value);
Packit 971217
      break;
Packit 971217
    case PROP_YPAD:
Packit 971217
      render->ypad = g_value_get_int (value);
Packit 971217
      break;
Packit 971217
    case PROP_FONT_DESC:
Packit 971217
    {
Packit 971217
      PangoFontDescription *desc;
Packit 971217
Packit 971217
      desc = pango_font_description_from_string (g_value_get_string (value));
Packit 971217
      if (desc) {
Packit 971217
        GST_LOG ("font description set: %s", g_value_get_string (value));
Packit 971217
        GST_OBJECT_LOCK (render);
Packit 971217
        pango_layout_set_font_description (render->layout, desc);
Packit 971217
        gst_text_render_adjust_values_with_fontdesc (render, desc);
Packit 971217
        pango_font_description_free (desc);
Packit 971217
        gst_text_render_render_pangocairo (render);
Packit 971217
        GST_OBJECT_UNLOCK (render);
Packit 971217
      } else {
Packit 971217
        GST_WARNING ("font description parse failed: %s",
Packit 971217
            g_value_get_string (value));
Packit 971217
      }
Packit 971217
      break;
Packit 971217
    }
Packit 971217
Packit 971217
    default:
Packit 971217
      break;
Packit 971217
  }
Packit 971217
}
Packit 971217
Packit 971217
static void
Packit 971217
gst_text_render_get_property (GObject * object, guint prop_id,
Packit 971217
    GValue * value, GParamSpec * pspec)
Packit 971217
{
Packit 971217
  GstTextRender *render = GST_TEXT_RENDER (object);
Packit 971217
Packit 971217
  switch (prop_id) {
Packit 971217
    case PROP_VALIGNMENT:
Packit 971217
      g_value_set_enum (value, render->valign);
Packit 971217
      break;
Packit 971217
    case PROP_HALIGNMENT:
Packit 971217
      g_value_set_enum (value, render->halign);
Packit 971217
      break;
Packit 971217
    case PROP_LINE_ALIGNMENT:
Packit 971217
      g_value_set_enum (value, render->line_align);
Packit 971217
      break;
Packit 971217
    case PROP_XPAD:
Packit 971217
      g_value_set_int (value, render->xpad);
Packit 971217
      break;
Packit 971217
    case PROP_YPAD:
Packit 971217
      g_value_set_int (value, render->ypad);
Packit 971217
      break;
Packit 971217
    default:
Packit 971217
      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit 971217
      break;
Packit 971217
  }
Packit 971217
}