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.
231 lines
8.1 KiB
231 lines
8.1 KiB
/*****************************************************************************
|
|
* encttml.c : TTML encoder
|
|
*****************************************************************************
|
|
* Copyright (C) 2018-2024 VideoLabs, VLC authors and VideoLAN
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation; either version 2.1 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <vlc_common.h>
|
|
#include <vlc_plugin.h>
|
|
#include <vlc_codec.h>
|
|
#include <vlc_subpicture.h>
|
|
|
|
#include "ttml.h"
|
|
|
|
#define HEX_COLOR_MAX 10
|
|
static void FillHexColor( uint32_t argb, bool withalpha, char text[HEX_COLOR_MAX] )
|
|
{
|
|
if( withalpha )
|
|
snprintf( text, HEX_COLOR_MAX, "#%08x", (argb << 8) | (argb >> 24) );
|
|
else
|
|
snprintf( text, HEX_COLOR_MAX, "#%06x", argb & 0x00FFFFFF );
|
|
}
|
|
|
|
static void AddTextNode( tt_node_t *p_parent, const char *psz_text )
|
|
{
|
|
const char *psz = psz_text;
|
|
const char *nl;
|
|
do
|
|
{
|
|
nl = strchr( psz, '\n' );
|
|
if( nl )
|
|
{
|
|
tt_subtextnode_New( p_parent, psz, nl - psz );
|
|
tt_node_New( p_parent, "br", NULL );
|
|
psz += nl - psz + 1;
|
|
if( *psz == '\0' )
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
tt_textnode_New( p_parent, psz );
|
|
}
|
|
} while ( nl );
|
|
}
|
|
|
|
static block_t *Encode( encoder_t *p_enc, subpicture_t *p_spu )
|
|
{
|
|
VLC_UNUSED( p_enc );
|
|
|
|
if( p_spu == NULL )
|
|
return NULL;
|
|
|
|
tt_node_t *p_root = tt_node_New( NULL, "tt", TT_NS );
|
|
if( !p_root )
|
|
return NULL;
|
|
|
|
tt_node_AddAttribute( p_root, "xmlns", TT_NS );
|
|
tt_node_AddAttribute( p_root, "xmlns:tts", TT_NS_STYLING );
|
|
|
|
tt_node_t *p_body = tt_node_New( p_root, "body", NULL );
|
|
if( !p_body )
|
|
{
|
|
tt_node_RecursiveDelete( p_root );
|
|
return NULL;
|
|
}
|
|
|
|
tt_node_t *p_div = tt_node_New( p_body, "div", NULL );
|
|
if( !p_div )
|
|
{
|
|
tt_node_RecursiveDelete( p_root );
|
|
return NULL;
|
|
}
|
|
|
|
subpicture_region_t *p_region;
|
|
vlc_spu_regions_foreach(p_region, &p_spu->regions)
|
|
{
|
|
if( !subpicture_region_IsText( p_region ) ||
|
|
p_region->p_text == NULL ||
|
|
p_region->p_text->psz_text == NULL )
|
|
continue;
|
|
|
|
tt_node_t *p_par = tt_node_New( p_div, "p", NULL );
|
|
if( !p_par )
|
|
continue;
|
|
|
|
if( p_spu->i_start != VLC_TICK_INVALID )
|
|
{
|
|
p_par->timings.begin = tt_time_Create( p_spu->i_start - VLC_TICK_0 );
|
|
if( p_spu->i_stop != VLC_TICK_INVALID && p_spu->i_stop > p_spu->i_start )
|
|
p_par->timings.end = tt_time_Create( p_spu->i_stop - VLC_TICK_0 );
|
|
tt_node_AddAttribute( p_par, "begin", "" );
|
|
}
|
|
|
|
for( const text_segment_t *p_segment = p_region->p_text;
|
|
p_segment; p_segment = p_segment->p_next )
|
|
{
|
|
if( p_segment->psz_text == NULL )
|
|
continue;
|
|
|
|
const text_style_t *style = p_segment->style;
|
|
if( style && style->i_features )
|
|
{
|
|
tt_node_t *p_span = tt_node_New( p_par, "span", NULL );
|
|
if( !p_span )
|
|
continue;
|
|
|
|
if( style->f_font_relsize && p_spu->i_original_picture_height )
|
|
{
|
|
char fontsize[10];
|
|
unsigned relem = p_spu->i_original_picture_height * style->f_font_relsize / 16;
|
|
snprintf( fontsize, 10, "%u%%", relem );
|
|
tt_node_AddAttribute( p_span, "tts:fontSize", fontsize );
|
|
}
|
|
else if ( style->i_font_size )
|
|
{
|
|
char fontsize[10];
|
|
snprintf( fontsize, 10, "%upx", style->i_font_size );
|
|
tt_node_AddAttribute( p_span, "tts:fontSize", fontsize );
|
|
}
|
|
|
|
if( style->psz_fontname )
|
|
tt_node_AddAttribute( p_span, "tts:fontFamily", style->psz_fontname );
|
|
|
|
if( style->i_features & STYLE_HAS_FLAGS )
|
|
{
|
|
if( style->i_style_flags & STYLE_BOLD )
|
|
tt_node_AddAttribute( p_span, "tts:fontWeight", "bold" );
|
|
if( style->i_style_flags & STYLE_ITALIC )
|
|
tt_node_AddAttribute( p_span, "tts:fontStyle", "italic" );
|
|
if( style->i_style_flags & STYLE_UNDERLINE )
|
|
tt_node_AddAttribute( p_span, "tts:textDecoration", "underline" );
|
|
if( style->i_style_flags & STYLE_STRIKEOUT )
|
|
tt_node_AddAttribute( p_span, "tts:textDecoration", "lineThrough" );
|
|
if( style->i_style_flags & STYLE_OUTLINE )
|
|
{
|
|
char color[HEX_COLOR_MAX];
|
|
uint32_t argb = style->i_outline_color;
|
|
if( style->i_features & STYLE_HAS_OUTLINE_ALPHA )
|
|
argb |= style->i_outline_alpha << 24;
|
|
FillHexColor( argb, style->i_features & STYLE_HAS_OUTLINE_ALPHA, color );
|
|
tt_node_AddAttribute( p_span, "tts:textOutline", color );
|
|
}
|
|
}
|
|
|
|
if( style->i_features & STYLE_HAS_FONT_COLOR )
|
|
{
|
|
char color[HEX_COLOR_MAX];
|
|
uint32_t argb = style->i_font_color;
|
|
if( style->i_features & STYLE_HAS_FONT_ALPHA )
|
|
argb |= style->i_font_alpha << 24;
|
|
FillHexColor( argb, style->i_features & STYLE_HAS_FONT_ALPHA, color );
|
|
tt_node_AddAttribute( p_span, "tts:color", color );
|
|
}
|
|
|
|
if( style->i_features & STYLE_HAS_BACKGROUND_COLOR )
|
|
{
|
|
char color[HEX_COLOR_MAX];
|
|
uint32_t argb = style->i_background_color;
|
|
if( style->i_features & STYLE_HAS_BACKGROUND_ALPHA )
|
|
argb |= style->i_background_alpha << 24;
|
|
FillHexColor( argb, style->i_features & STYLE_HAS_BACKGROUND_ALPHA, color );
|
|
tt_node_AddAttribute( p_span, "tts:backgroundColor", color );
|
|
}
|
|
|
|
AddTextNode( p_span, p_segment->psz_text );
|
|
}
|
|
else
|
|
{
|
|
AddTextNode( p_par, p_segment->psz_text );
|
|
}
|
|
}
|
|
}
|
|
|
|
block_t* p_block = NULL;
|
|
struct vlc_memstream stream;
|
|
|
|
if( !vlc_memstream_open( &stream ) )
|
|
{
|
|
tt_time_t playbacktime = tt_time_Create( p_spu->i_start );
|
|
|
|
tt_node_ToText( &stream, (tt_basenode_t *)p_root, &playbacktime );
|
|
if( !vlc_memstream_close( &stream ) )
|
|
{
|
|
p_block = block_heap_Alloc( stream.ptr, stream.length );
|
|
if( p_block )
|
|
{
|
|
p_block->i_dts = p_block->i_pts = VLC_TICK_0 + p_spu->i_start;
|
|
if( p_spu->i_stop != VLC_TICK_INVALID && p_spu->i_stop > p_spu->i_start )
|
|
p_block->i_length = p_spu->i_stop - p_spu->i_start;
|
|
}
|
|
}
|
|
}
|
|
|
|
tt_node_RecursiveDelete( p_root );
|
|
|
|
return p_block;
|
|
}
|
|
|
|
int tt_OpenEncoder( vlc_object_t *p_this )
|
|
{
|
|
encoder_t *p_enc = (encoder_t *)p_this;
|
|
|
|
if( p_enc->fmt_out.i_codec != VLC_CODEC_TTML )
|
|
return VLC_EGENERIC;
|
|
|
|
p_enc->p_sys = NULL;
|
|
|
|
static const struct vlc_encoder_operations ops =
|
|
{ .encode_sub = Encode };
|
|
p_enc->ops = &ops;
|
|
|
|
p_enc->fmt_out.i_cat = SPU_ES;
|
|
return VLC_SUCCESS;
|
|
}
|
|
|