time.c 12.6 KB
Newer Older
1 2 3
/*****************************************************************************
 * time.c : time display video plugin for vlc
 *****************************************************************************
4
 * Copyright (C) 2003-2005 VideoLAN
5
 * $Id$
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
 *
 * Authors: Sigmund Augdal <sigmunau@idi.ntnu.no>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 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 General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
 *****************************************************************************/

/*****************************************************************************
 * Preamble
 *****************************************************************************/
#include <stdlib.h>                                      /* malloc(), free() */
#include <string.h>

#include <vlc/vlc.h>
#include <vlc/vout.h>

#include "vlc_filter.h"
#include "vlc_block.h"
35
#include "osd.h"
36 37 38 39 40 41 42

/*****************************************************************************
 * Local prototypes
 *****************************************************************************/
static int  CreateFilter ( vlc_object_t * );
static void DestroyFilter( vlc_object_t * );
static subpicture_t *Filter( filter_t *, mtime_t );
43 44 45 46 47 48 49 50 51 52 53 54
static int TimeCallback( vlc_object_t *p_this, char const *psz_var,
                            vlc_value_t oldval, vlc_value_t newval,
                            void *p_data );
static int pi_color_values[] = { 0xf0000000, 0x00000000, 0x00808080, 0x00C0C0C0, 
               0x00FFFFFF, 0x00800000, 0x00FF0000, 0x00FF00FF, 0x00FFFF00, 
               0x00808000, 0x00008000, 0x00008080, 0x0000FF00, 0x00800080, 
               0x00000080, 0x000000FF, 0x0000FFFF}; 
static char *ppsz_color_descriptions[] = { N_("Default"), N_("Black"), 
               N_("Gray"), N_("Silver"), N_("White"), N_("Maroon"), N_("Red"),
               N_("Fuchsia"), N_("Yellow"), N_("Olive"), N_("Green"), 
               N_("Teal"), N_("Lime"), N_("Purple"), N_("Navy"), N_("Blue"), 
               N_("Aqua") };
55 56 57 58 59 60

/*****************************************************************************
 * filter_sys_t: time filter descriptor
 *****************************************************************************/
struct filter_sys_t
{
61
    int i_xoff, i_yoff;  /* offsets for the display string in the video window */
62
    char *psz_format;    /* time format string */
63
    int i_pos;  /* permit relative positioning (top, bottom, left, right, center) */
64
    int  i_font_color, i_font_opacity, i_font_size; /* font control */
65 66

    time_t last_time;
67 68
};

69 70 71 72 73 74
#define MSG_TEXT N_("Time format string (%Y%m%d %H%M%S)")
#define MSG_LONGTEXT N_("Time format string (%Y = year, %m = month, %d = day, %H = hour, %M = minute, %S = second")
#define POSX_TEXT N_("X offset, from left")
#define POSX_LONGTEXT N_("X offset, from the left screen edge" )
#define POSY_TEXT N_("Y offset, from the top")
#define POSY_LONGTEXT N_("Y offset, down from the top" )
75 76 77
#define OPACITY_TEXT N_("Opacity, -1..255")
#define OPACITY_LONGTEXT N_("The opacity (inverse of transparency) of overlay text. " \
    "-1 = use freetype-opacity, 0 = transparent, 255 = totally opaque. " )
78 79 80 81
#define SIZE_TEXT N_("Font size, pixels")
#define SIZE_LONGTEXT N_("Specify the font size, in pixels, " \
    "with -1 = use freetype-fontsize" )

82 83 84 85 86 87 88 89 90 91 92 93 94 95
#define COLOR_TEXT N_("Text Default Color")
#define COLOR_LONGTEXT N_("The color of overlay text. 1 byte for each color, hexadecimal." \
    "-1 = use freetype-color, #000000 = all colors off, " \
    "0xFF0000 = just Red, 0xFFFFFF = all color on [White]" )
#define POS_TEXT N_("Time position")
#define POS_LONGTEXT N_( \
  "You can enforce the time position on the video " \
  "(0=center, 1=left, 2=right, 4=top, 8=bottom, you can " \
  "also use combinations of these values by adding them).")

static int pi_pos_values[] = { 0, 1, 2, 4, 8, 5, 6, 9, 10 };
static char *ppsz_pos_descriptions[] =
     { N_("Center"), N_("Left"), N_("Right"), N_("Top"), N_("Bottom"),
     N_("Top-Left"), N_("Top-Right"), N_("Bottom-Left"), N_("Bottom-Right") };
96 97 98 99 100 101

/*****************************************************************************
 * Module descriptor
 *****************************************************************************/
vlc_module_begin();
    set_capability( "sub filter", 0 );
zorglub's avatar
 
zorglub committed
102
    set_shortname( N_("Time overlay"));
zorglub's avatar
zorglub committed
103 104
    set_category( CAT_VIDEO );
    set_subcategory( SUBCAT_VIDEO_SUBPIC );
105
    set_callbacks( CreateFilter, DestroyFilter );
106
    add_string( "time-format", "%Y-%m-%d   %H:%M:%S", NULL, MSG_TEXT, MSG_LONGTEXT, VLC_FALSE );
107
    add_integer( "time-x", -1, NULL, POSX_TEXT, POSX_LONGTEXT, VLC_FALSE );
108
    add_integer( "time-y", 0, NULL, POSY_TEXT, POSY_LONGTEXT, VLC_FALSE );
109 110 111 112 113 114 115
    add_integer( "time-position", 9, NULL, POS_TEXT, POS_LONGTEXT, VLC_TRUE );
    /* 9 sets the default to bottom-left, minimizing jitter */
    change_integer_list( pi_pos_values, ppsz_pos_descriptions, 0 );
    add_integer_with_range( "time-opacity", -1, -1, 255, NULL,
        OPACITY_TEXT, OPACITY_LONGTEXT, VLC_FALSE );
    add_integer( "time-color", -1, NULL, COLOR_TEXT, COLOR_LONGTEXT, VLC_TRUE );
        change_integer_list( pi_color_values, ppsz_color_descriptions, 0 );
116
    add_integer( "time-size", -1, NULL, SIZE_TEXT, SIZE_LONGTEXT, VLC_FALSE );
117 118
    set_description( _("Time display sub filter") );
    add_shortcut( "time" );
119 120 121
vlc_module_end();

/*****************************************************************************
122
 * CreateFilter: allocates time video filter
123 124 125 126 127
 *****************************************************************************/
static int CreateFilter( vlc_object_t *p_this )
{
    filter_t *p_filter = (filter_t *)p_this;
    filter_sys_t *p_sys;
128
    vlc_object_t *p_input;
129 130 131 132 133 134 135 136

    /* Allocate structure */
    p_sys = p_filter->p_sys = malloc( sizeof( filter_sys_t ) );
    if( p_sys == NULL )
    {
        msg_Err( p_filter, "out of memory" );
        return VLC_ENOMEM;
    }
137 138 139 140 141 142 143 144 145 146 147 148 149 150
    /* Hook used for callback variables */
    p_input = vlc_object_find( p_this, VLC_OBJECT_INPUT, FIND_ANYWHERE );
    if( !p_input )
    {
        return VLC_ENOOBJ;
    }

    p_sys->i_xoff = var_CreateGetInteger( p_input->p_libvlc , "time-x" );
    p_sys->i_yoff = var_CreateGetInteger( p_input->p_libvlc , "time-y" );
    p_sys->psz_format = var_CreateGetString( p_input->p_libvlc, "time-format" );
    p_sys->i_pos = var_CreateGetInteger( p_input->p_libvlc , "time-position" );
    var_Create( p_input->p_libvlc, "time-opacity", VLC_VAR_INTEGER|VLC_VAR_DOINHERIT );
    p_sys->i_font_opacity = var_CreateGetInteger( p_input->p_libvlc , "time-opacity" );
    p_sys->i_font_color = var_CreateGetInteger( p_input->p_libvlc , "time-color" );
151 152
    p_sys->i_font_size = var_CreateGetInteger( p_input->p_libvlc , "time-size" );
   
153 154 155 156 157 158
    var_AddCallback( p_input->p_libvlc, "time-x", TimeCallback, p_sys );
    var_AddCallback( p_input->p_libvlc, "time-y", TimeCallback, p_sys );
    var_AddCallback( p_input->p_libvlc, "time-format", TimeCallback, p_sys );
    var_AddCallback( p_input->p_libvlc, "time-position", TimeCallback, p_sys );
    var_AddCallback( p_input->p_libvlc, "time-color", TimeCallback, p_sys );
    var_AddCallback( p_input->p_libvlc, "time-opacity", TimeCallback, p_sys );
159
    var_AddCallback( p_input->p_libvlc, "time-size", TimeCallback, p_sys );
160

161
    vlc_object_release( p_input );
162

163 164
    /* Misc init */
    p_filter->pf_sub_filter = Filter;
165
    p_sys->last_time = ((time_t)-1);
166 167 168 169 170 171 172 173 174 175

    return VLC_SUCCESS;
}
/*****************************************************************************
 * DestroyFilter: destroy logo video filter
 *****************************************************************************/
static void DestroyFilter( vlc_object_t *p_this )
{
    filter_t *p_filter = (filter_t *)p_this;
    filter_sys_t *p_sys = p_filter->p_sys;
176
    vlc_object_t *p_input;
177

178
    if( p_sys->psz_format ) free( p_sys->psz_format );
179
    free( p_sys );
180 181 182 183 184 185 186 187 188 189 190 191
    /* Delete the time variables */
    p_input = vlc_object_find( p_this, VLC_OBJECT_INPUT, FIND_ANYWHERE );
    if( !p_input )
    {
        return;
    }
    var_Destroy( p_input->p_libvlc , "time-format" );
    var_Destroy( p_input->p_libvlc , "time-x" );
    var_Destroy( p_input->p_libvlc , "time-y" );
    var_Destroy( p_input->p_libvlc , "time-position" );
    var_Destroy( p_input->p_libvlc , "time-color");
    var_Destroy( p_input->p_libvlc , "time-opacity");
192 193
    var_Destroy( p_input->p_libvlc , "time-size");
   
194
    vlc_object_release( p_input );
195 196
}

197
static char *FormatTime(char *tformat )
198
{
199 200 201 202 203 204 205 206 207 208 209 210 211 212 213
  char buffer[255];
  time_t curtime;
#if defined(HAVE_LOCALTIME_R)
  struct tm loctime;
#else
  struct tm *loctime;
#endif

  /* Get the current time.  */
  curtime = time( NULL );

  /* Convert it to local time representation.  */
#if defined(HAVE_LOCALTIME_R)
  localtime_r( &curtime, &loctime );
  strftime( buffer, 255, tformat, &loctime );
214
#else
215 216
  loctime = localtime( &curtime );
  strftime( buffer, 255, tformat, loctime );
217
#endif
218
  return strdup( buffer );
219 220 221 222 223 224 225 226 227
}

/****************************************************************************
 * Filter: the whole thing
 ****************************************************************************
 * This function outputs subpictures at regular time intervals.
 ****************************************************************************/
static subpicture_t *Filter( filter_t *p_filter, mtime_t date )
{
228
    filter_sys_t *p_sys = p_filter->p_sys;
229 230
    subpicture_t *p_spu;
    video_format_t fmt;
231

232 233
    if( p_sys->last_time == time( NULL ) ) return NULL;

234 235
    p_spu = p_filter->pf_sub_buffer_new( p_filter );
    if( !p_spu ) return NULL;
236

237 238 239
    memset( &fmt, 0, sizeof(video_format_t) );
    fmt.i_chroma = VLC_FOURCC('T','E','X','T');
    fmt.i_aspect = 0;
240 241 242
    fmt.i_width = fmt.i_height = 0;     
    fmt.i_x_offset = 0;
    fmt.i_y_offset = 0;
243 244 245 246 247 248
    p_spu->p_region = p_spu->pf_create_region( VLC_OBJECT(p_filter), &fmt );
    if( !p_spu->p_region )
    {
        p_filter->pf_sub_buffer_del( p_filter, p_spu );
        return NULL;
    }
249

250
    p_sys->last_time = time( NULL );
251

252
    p_spu->p_region->psz_text = FormatTime( p_sys->psz_format );
253
    p_spu->i_start = date;
254
    p_spu->i_stop  = 0;
255
    p_spu->b_ephemer = VLC_TRUE;
256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271

    /*  where to locate the string: */
    if( p_sys->i_xoff < 0 || p_sys->i_yoff < 0 )
    {   /* set to one of the 9 relative locations */
        p_spu->i_flags = p_sys->i_pos;
        p_spu->i_x = 0;
        p_spu->i_y = 0;
        p_spu->b_absolute = VLC_FALSE;
    }
    else
    {   /*  set to an absolute xy, referenced to upper left corner */
	    p_spu->i_flags = OSD_ALIGN_LEFT | OSD_ALIGN_TOP;
        p_spu->i_x = p_sys->i_xoff;
        p_spu->i_y = p_sys->i_yoff;
        p_spu->b_absolute = VLC_TRUE;
    }
272
    p_spu->p_region->i_font_color = p_sys->i_font_color;
273 274
    p_spu->p_region->i_font_opacity = p_sys->i_font_opacity;
    p_spu->p_region->i_font_size = p_sys->i_font_size;
275

276 277
    return p_spu;
}
278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299
/**********************************************************************
 * Callback to update params on the fly
 **********************************************************************/
static int TimeCallback( vlc_object_t *p_this, char const *psz_var,
                            vlc_value_t oldval, vlc_value_t newval,
                            void *p_data )
{
    filter_sys_t *p_sys = (filter_sys_t *) p_data;

    if( !strncmp( psz_var, "time-format", 11 ) )
    {
        if( p_sys->psz_format ) free( p_sys->psz_format );
        p_sys->psz_format = strdup( newval.psz_string );
    }
    else if ( !strncmp( psz_var, "time-x", 6 ) )
    {
        p_sys->i_xoff = newval.i_int;
    }
    else if ( !strncmp( psz_var, "time-y", 6 ) )
    {
        p_sys->i_yoff = newval.i_int;
    }
300
    else if ( !strncmp( psz_var, "time-color", 8 ) )  /* "time-c" */ 
301 302 303
    {
        p_sys->i_font_color = newval.i_int;
    }
304
    else if ( !strncmp( psz_var, "time-opacity", 8 ) ) /* "time-o" */ 
305 306 307
    {
        p_sys->i_font_opacity = newval.i_int;
    }
308 309 310 311
    else if ( !strncmp( psz_var, "time-size", 6 ) )
    {
        p_sys->i_font_size = newval.i_int;
    }
312 313 314 315 316 317 318 319
    else if ( !strncmp( psz_var, "time-position", 8 ) )
    /* willing to accept a match against time-pos */
    {
        p_sys->i_pos = newval.i_int;
        p_sys->i_xoff = -1;       /* force to relative positioning */
    }
    return VLC_SUCCESS;
}