/* * Copyright (C) 2006 Evgeniy Stepanov * * This file is part of MPlayer. * * MPlayer 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. * * MPlayer 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 libass; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include #include #include #include #include #include "config.h" #include "mp_msg.h" #include "path.h" #include "ass_mp.h" #include "subreader.h" #include "sub/sub.h" #include "stream/stream.h" #include "options.h" ASS_Track *mp_ass_default_track(ASS_Library *library, struct MPOpts *opts) { ASS_Track *track = ass_new_track(library); track->track_type = TRACK_TYPE_ASS; track->Timer = 100.; track->PlayResY = 288; track->WrapStyle = 0; if (opts->ass_styles_file) ass_read_styles(track, opts->ass_styles_file, sub_cp); if (track->n_styles == 0) { track->Kerning = true; int sid = ass_alloc_style(track); track->default_style = sid; ASS_Style *style = track->styles + sid; style->Name = strdup("Default"); style->FontName = sub_font_name ? strdup(sub_font_name) : font_name ? strdup(font_name) : strdup("Sans"); style->treat_fontname_as_pattern = 1; double fs = track->PlayResY * text_font_scale_factor / 100.; /* The font size is always proportional to video height only; * real -subfont-autoscale behavior is not implemented. * Apply a correction that corresponds to about 4:3 aspect ratio * video to get a size somewhat closer to what non-libass rendering * would produce with the same text_font_scale_factor * and subtitle_autoscale. */ if (subtitle_autoscale == 2) fs *= 1.3; else if (subtitle_autoscale == 3) fs *= 1.7; uint32_t c1 = 0xFFFFFF00; uint32_t c2 = 0x00000000; if (opts->ass_color) c1 = strtoll(opts->ass_color, NULL, 16); if (opts->ass_border_color) c2 = strtoll(opts->ass_border_color, NULL, 16); style->FontSize = fs; style->PrimaryColour = c1; style->SecondaryColour = c1; style->OutlineColour = c2; style->BackColour = 0x00000000; style->BorderStyle = 1; style->Alignment = 2; style->Outline = fs / 16; style->MarginL = 10; style->MarginR = 10; style->MarginV = 5; style->ScaleX = 1.; style->ScaleY = 1.; } ass_process_force_style(track); return track; } static int check_duplicate_plaintext_event(ASS_Track *track) { int i; ASS_Event *evt = track->events + track->n_events - 1; for (i = 0; i < track->n_events - 1; ++i) // ignoring last event, it is the one we are comparing with if (track->events[i].Start == evt->Start && track->events[i].Duration == evt->Duration && strcmp(track->events[i].Text, evt->Text) == 0) return 1; return 0; } /** * \brief Convert subtitle to ASS_Events for the given track * \param track track * \param sub subtitle to convert * \return event id * note: assumes that subtitle is _not_ fps-based; caller must manually correct * Start and Duration in other case. **/ static int ass_process_subtitle(ASS_Track *track, subtitle *sub) { int eid; ASS_Event *event; int len = 0, j; char *p; char *end; eid = ass_alloc_event(track); event = track->events + eid; event->Start = sub->start * 10; event->Duration = (sub->end - sub->start) * 10; event->Style = track->default_style; for (j = 0; j < sub->lines; ++j) len += sub->text[j] ? strlen(sub->text[j]) : 0; len += 2 * sub->lines; // '\N', including the one after the last line len += 6; // {\anX} len += 1; // '\0' event->Text = malloc(len); end = event->Text + len; p = event->Text; if (sub->alignment) p += snprintf(p, end - p, "{\\an%d}", sub->alignment); for (j = 0; j < sub->lines; ++j) p += snprintf(p, end - p, "%s\\N", sub->text[j]); if (sub->lines > 0) p -= 2; // remove last "\N" *p = 0; if (check_duplicate_plaintext_event(track)) { ass_free_event(track, eid); track->n_events--; return -1; } mp_msg(MSGT_ASS, MSGL_V, "plaintext event at %" PRId64 ", +%" PRId64 ": %s \n", (int64_t) event->Start, (int64_t) event->Duration, event->Text); return eid; } /** * \brief Convert subdata to ASS_Track * \param subdata subtitles struct from subreader * \param fps video framerate * \return newly allocated ASS_Track, filled with subtitles from subdata */ ASS_Track *mp_ass_read_subdata(ASS_Library *library, struct MPOpts *opts, sub_data *subdata, double fps) { ASS_Track *track; int i; track = mp_ass_default_track(library, opts); track->name = subdata->filename ? strdup(subdata->filename) : 0; for (i = 0; i < subdata->sub_num; ++i) { int eid = ass_process_subtitle(track, subdata->subtitles + i); if (eid < 0) continue; if (!subdata->sub_uses_time) { track->events[eid].Start *= 100. / fps; track->events[eid].Duration *= 100. / fps; } } return track; } ASS_Track *mp_ass_read_stream(ASS_Library *library, const char *fname, char *charset) { ASS_Track *track; struct stream *s = open_stream(fname, NULL, NULL); if (!s) // Stream code should have printed an error already return NULL; struct bstr content = stream_read_complete(s, NULL, 100000000, 1); if (content.start == NULL) mp_tmsg(MSGT_ASS, MSGL_ERR, "Refusing to load subtitle file " "larger than 100 MB: %s\n", fname); free_stream(s); if (content.len == 0) { talloc_free(content.start); return NULL; } content.start[content.len] = 0; track = ass_read_memory(library, content.start, content.len, charset); if (track) { free(track->name); track->name = strdup(fname); } talloc_free(content.start); return track; } void mp_ass_configure(ASS_Renderer *priv, struct MPOpts *opts, int w, int h, bool unscaled) { int hinting; ass_set_frame_size(priv, w, h); ass_set_margins(priv, opts->ass_top_margin, opts->ass_bottom_margin, 0, 0); mp_ass_reload_options(priv, opts); if (!unscaled && (opts->ass_hinting & 4)) hinting = 0; else hinting = opts->ass_hinting & 3; ass_set_hinting(priv, hinting); ass_set_line_spacing(priv, opts->ass_line_spacing); } void mp_ass_configure_fonts(ASS_Renderer *priv) { char *dir, *path, *family; dir = get_path("fonts"); path = get_path("subfont.ttf"); if (!mp_path_exists(path)) { free(path); path = NULL; } if (sub_font_name) family = strdup(sub_font_name); else if (font_name) family = strdup(font_name); else family = 0; ass_set_fonts(priv, path, family, 1, NULL, 1); free(dir); free(path); free(family); } static int map_ass_level[] = { MSGL_ERR, // 0 "FATAL errors" MSGL_WARN, MSGL_INFO, MSGL_V, MSGL_V, MSGL_V, // 5 application recommended level MSGL_DBG2, MSGL_DBG3, // 7 "verbose DEBUG" }; static void message_callback(int level, const char *format, va_list va, void *ctx) { level = map_ass_level[level]; mp_msg(MSGT_ASS, level, "[ass] "); mp_msg_va(MSGT_ASS, level, format, va); // libass messages lack trailing \n mp_msg(MSGT_ASS, level, "\n"); } ASS_Library *mp_ass_init(struct MPOpts *opts) { ASS_Library *priv; char *path = get_path("fonts"); priv = ass_library_init(); ass_set_message_cb(priv, message_callback, NULL); ass_set_fonts_dir(priv, path); ass_set_extract_fonts(priv, opts->use_embedded_fonts); free(path); return priv; } void mp_ass_reload_options(ASS_Renderer *priv, struct MPOpts *opts) { /* This could be needed for vf_ass case if the margins were actually * runtime configurable, but would be wrong with EOSD: * ass_set_margins(priv, opts->ass_top_margin, opts->ass_bottom_margin, * 0, 0); */ ass_set_use_margins(priv, opts->ass_use_margins); ass_set_font_scale(priv, opts->ass_font_scale); }