2 * Copyright (C) 2006 Evgeniy Stepanov <eugeni.stepanov@gmail.com>
4 * This file is part of MPlayer.
6 * MPlayer is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * MPlayer is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with libass; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
28 #include <ass/ass_types.h>
30 #include <libavutil/common.h>
36 #include "subreader.h"
37 #include "stream/stream.h"
39 #ifdef CONFIG_FONTCONFIG
40 #include <fontconfig/fontconfig.h>
43 // libass-related command line options
44 ASS_Library
*ass_library
;
45 float ass_font_scale
= 1.;
46 float ass_line_spacing
= 0.;
47 int ass_top_margin
= 0;
48 int ass_bottom_margin
= 0;
49 int use_embedded_fonts
= 1;
50 char **ass_force_style_list
= NULL
;
51 int ass_use_margins
= 0;
52 char *ass_color
= NULL
;
53 char *ass_border_color
= NULL
;
54 char *ass_styles_file
= NULL
;
55 int ass_hinting
= ASS_HINTING_LIGHT
+ 4; // light hinting for unscaled osd
57 #ifdef CONFIG_FONTCONFIG
58 extern int font_fontconfig
;
60 static int font_fontconfig
= -1;
62 extern char *font_name
;
63 extern char *sub_font_name
;
64 extern float text_font_scale_factor
;
65 extern int subtitle_autoscale
;
70 static char *sub_cp
= 0;
73 void process_force_style(ASS_Track
*track
);
75 ASS_Track
*ass_default_track(ASS_Library
*library
)
77 ASS_Track
*track
= ass_new_track(library
);
79 track
->track_type
= TRACK_TYPE_ASS
;
81 track
->PlayResY
= 288;
85 ass_read_styles(track
, ass_styles_file
, sub_cp
);
87 if (track
->n_styles
== 0) {
93 sid
= ass_alloc_style(track
);
94 style
= track
->styles
+ sid
;
95 style
->Name
= strdup("Default");
96 style
->FontName
= (font_fontconfig
>= 0
97 && sub_font_name
) ? strdup(sub_font_name
)
98 : (font_fontconfig
>= 0
99 && font_name
) ? strdup(font_name
) : strdup("Sans");
100 style
->treat_fontname_as_pattern
= 1;
102 fs
= track
->PlayResY
* text_font_scale_factor
/ 100.;
103 // approximate autoscale coefficients
104 if (subtitle_autoscale
== 2)
106 else if (subtitle_autoscale
== 3)
108 style
->FontSize
= fs
;
111 c1
= strtoll(ass_color
, NULL
, 16);
114 if (ass_border_color
)
115 c2
= strtoll(ass_border_color
, NULL
, 16);
119 style
->PrimaryColour
= c1
;
120 style
->SecondaryColour
= c1
;
121 style
->OutlineColour
= c2
;
122 style
->BackColour
= 0x00000000;
123 style
->BorderStyle
= 1;
124 style
->Alignment
= 2;
133 ass_process_force_style(track
);
137 static int check_duplicate_plaintext_event(ASS_Track
*track
)
140 ASS_Event
*evt
= track
->events
+ track
->n_events
- 1;
142 for (i
= 0; i
< track
->n_events
- 1; ++i
) // ignoring last event, it is the one we are comparing with
143 if (track
->events
[i
].Start
== evt
->Start
&&
144 track
->events
[i
].Duration
== evt
->Duration
&&
145 strcmp(track
->events
[i
].Text
, evt
->Text
) == 0)
151 * \brief Convert subtitle to ASS_Events for the given track
153 * \param sub subtitle to convert
155 * note: assumes that subtitle is _not_ fps-based; caller must manually correct
156 * Start and Duration in other case.
158 static int ass_process_subtitle(ASS_Track
*track
, subtitle
*sub
)
166 eid
= ass_alloc_event(track
);
167 event
= track
->events
+ eid
;
169 event
->Start
= sub
->start
* 10;
170 event
->Duration
= (sub
->end
- sub
->start
) * 10;
173 for (j
= 0; j
< sub
->lines
; ++j
)
174 len
+= sub
->text
[j
] ? strlen(sub
->text
[j
]) : 0;
176 len
+= 2 * sub
->lines
; // '\N', including the one after the last line
180 event
->Text
= malloc(len
);
181 end
= event
->Text
+ len
;
185 p
+= snprintf(p
, end
- p
, "{\\an%d}", sub
->alignment
);
187 for (j
= 0; j
< sub
->lines
; ++j
)
188 p
+= snprintf(p
, end
- p
, "%s\\N", sub
->text
[j
]);
191 p
-= 2; // remove last "\N"
194 if (check_duplicate_plaintext_event(track
)) {
195 ass_free_event(track
, eid
);
200 mp_msg(MSGT_ASS
, MSGL_V
,
201 "plaintext event at %" PRId64
", +%" PRId64
": %s \n",
202 (int64_t) event
->Start
, (int64_t) event
->Duration
, event
->Text
);
209 * \brief Convert subdata to ASS_Track
210 * \param subdata subtitles struct from subreader
211 * \param fps video framerate
212 * \return newly allocated ASS_Track, filled with subtitles from subdata
214 ASS_Track
*ass_read_subdata(ASS_Library
*library
, sub_data
*subdata
,
220 track
= ass_default_track(library
);
221 track
->name
= subdata
->filename
? strdup(subdata
->filename
) : 0;
223 for (i
= 0; i
< subdata
->sub_num
; ++i
) {
224 int eid
= ass_process_subtitle(track
, subdata
->subtitles
+ i
);
227 if (!subdata
->sub_uses_time
) {
228 track
->events
[eid
].Start
*= 100. / fps
;
229 track
->events
[eid
].Duration
*= 100. / fps
;
235 ASS_Track
*ass_read_stream(ASS_Library
*library
, const char *fname
, char *charset
)
241 size_t buf_alloc
= 0;
244 fd
= open_stream(fname
, NULL
, NULL
);
246 // Stream code should have printed an error already
248 if (fd
->end_pos
> STREAM_BUFFER_SIZE
)
249 /* read entire file if size is known */
250 buf_alloc
= fd
->end_pos
;
254 if (sz
> 100000000) {
255 mp_tmsg(MSGT_ASS
, MSGL_ERR
, "Refusing to load subtitle file "
256 "larger than 100 MB: %s\n", fname
);
260 buf_alloc
= FFMAX(buf_alloc
, sz
+ (sz
>> 1));
261 buf_alloc
= FFMIN(buf_alloc
, 100000001);
262 buf
= realloc(buf
, buf_alloc
+ 1);
263 i
= stream_read(fd
, buf
+ sz
, buf_alloc
- sz
);
274 buf
= realloc(buf
, sz
+ 1);
275 track
= ass_read_memory(library
, buf
, sz
, charset
);
278 track
->name
= strdup(fname
);
284 void ass_configure(ASS_Renderer
*priv
, int w
, int h
, bool unscaled
)
287 ass_set_frame_size(priv
, w
, h
);
288 ass_set_margins(priv
, ass_top_margin
, ass_bottom_margin
, 0, 0);
289 ass_set_use_margins(priv
, ass_use_margins
);
290 ass_set_font_scale(priv
, ass_font_scale
);
291 if (!unscaled
&& (ass_hinting
& 4))
294 hinting
= ass_hinting
& 3;
295 ass_set_hinting(priv
, hinting
);
296 ass_set_line_spacing(priv
, ass_line_spacing
);
299 void ass_configure_fonts(ASS_Renderer
*priv
)
301 char *dir
, *path
, *family
;
302 dir
= get_path("fonts");
303 if (font_fontconfig
< 0 && sub_font_name
)
304 path
= strdup(sub_font_name
);
305 else if (font_fontconfig
< 0 && font_name
)
306 path
= strdup(font_name
);
308 path
= get_path("subfont.ttf");
309 if (font_fontconfig
>= 0 && sub_font_name
)
310 family
= strdup(sub_font_name
);
311 else if (font_fontconfig
>= 0 && font_name
)
312 family
= strdup(font_name
);
316 ass_set_fonts(priv
, path
, family
, font_fontconfig
+ 1, NULL
, 1);
323 static void message_callback(int level
, const char *format
, va_list va
, void *ctx
)
325 mp_msg(MSGT_ASS
, level
, "[ass] ");
326 mp_msg_va(MSGT_ASS
, level
, format
, va
);
327 // libass messages lack trailing \n
328 mp_msg(MSGT_ASS
, level
, "\n");
331 ASS_Library
*ass_init(void)
334 char *path
= get_path("fonts");
335 priv
= ass_library_init();
336 ass_set_message_cb(priv
, message_callback
, NULL
);
337 ass_set_fonts_dir(priv
, path
);
338 ass_set_extract_fonts(priv
, use_embedded_fonts
);
339 ass_set_style_overrides(priv
, ass_force_style_list
);
344 int ass_force_reload
= 0; // flag set if global ass-related settings were changed
346 ASS_Image
*ass_mp_render_frame(ASS_Renderer
*priv
, ASS_Track
*track
,
347 long long now
, int *detect_change
)
349 if (ass_force_reload
) {
350 ass_set_margins(priv
, ass_top_margin
, ass_bottom_margin
, 0, 0);
351 ass_set_use_margins(priv
, ass_use_margins
);
352 ass_set_font_scale(priv
, ass_font_scale
);
353 ass_force_reload
= 0;
355 return ass_render_frame(priv
, track
, now
, detect_change
);