2011-01-16 18:03:08 +00:00
|
|
|
/*
|
|
|
|
* 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 MPlayer; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <ass/ass.h>
|
|
|
|
#include <assert.h>
|
2011-01-18 13:33:36 +00:00
|
|
|
#include <string.h>
|
2011-01-16 18:03:08 +00:00
|
|
|
|
|
|
|
#include "talloc.h"
|
|
|
|
|
|
|
|
#include "mpcommon.h"
|
|
|
|
#include "libmpdemux/stheader.h"
|
2011-01-26 17:40:52 +00:00
|
|
|
#include "sub.h"
|
2011-01-16 18:03:08 +00:00
|
|
|
#include "ass_mp.h"
|
|
|
|
#include "sd.h"
|
2011-01-18 13:33:36 +00:00
|
|
|
#include "subassconvert.h"
|
2011-01-16 18:03:08 +00:00
|
|
|
|
|
|
|
struct sd_ass_priv {
|
|
|
|
struct ass_track *ass_track;
|
2011-01-18 13:33:36 +00:00
|
|
|
bool incomplete_event;
|
2011-01-16 18:03:08 +00:00
|
|
|
};
|
|
|
|
|
2011-01-18 13:33:36 +00:00
|
|
|
static void free_last_event(ASS_Track *track)
|
|
|
|
{
|
|
|
|
assert(track->n_events > 0);
|
|
|
|
ass_free_event(track, track->n_events - 1);
|
|
|
|
track->n_events--;
|
|
|
|
}
|
|
|
|
|
2011-01-16 18:03:08 +00:00
|
|
|
static void init(struct sh_sub *sh, struct osd_state *osd)
|
|
|
|
{
|
|
|
|
struct sd_ass_priv *ctx;
|
|
|
|
|
|
|
|
if (sh->initialized) {
|
|
|
|
ctx = sh->context;
|
|
|
|
} else {
|
|
|
|
ctx = talloc_zero(NULL, struct sd_ass_priv);
|
|
|
|
sh->context = ctx;
|
|
|
|
if (sh->type == 'a') {
|
|
|
|
ctx->ass_track = ass_new_track(ass_library);
|
|
|
|
if (sh->extradata)
|
|
|
|
ass_process_codec_private(ctx->ass_track, sh->extradata,
|
|
|
|
sh->extradata_len);
|
|
|
|
} else
|
2011-01-20 16:47:18 +00:00
|
|
|
ctx->ass_track = mp_ass_default_track(ass_library);
|
2011-01-16 18:03:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(osd->ass_track == NULL);
|
|
|
|
osd->ass_track = ctx->ass_track;
|
2011-01-19 18:13:48 +00:00
|
|
|
osd->vsfilter_aspect = sh->type == 'a';
|
|
|
|
osd->ass_track_changed = true;
|
2011-01-16 18:03:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void decode(struct sh_sub *sh, struct osd_state *osd, void *data,
|
|
|
|
int data_len, double pts, double duration)
|
|
|
|
{
|
2011-01-18 13:33:36 +00:00
|
|
|
unsigned char *text = data;
|
2011-01-16 18:03:08 +00:00
|
|
|
struct sd_ass_priv *ctx = sh->context;
|
2011-01-18 13:33:36 +00:00
|
|
|
ASS_Track *track = ctx->ass_track;
|
2011-01-16 18:03:08 +00:00
|
|
|
|
|
|
|
if (sh->type == 'a') { // ssa/ass subs
|
2011-01-18 13:33:36 +00:00
|
|
|
ass_process_chunk(track, data, data_len,
|
2011-01-16 18:03:08 +00:00
|
|
|
(long long)(pts*1000 + 0.5),
|
|
|
|
(long long)(duration*1000 + 0.5));
|
2011-01-18 13:33:36 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// plaintext subs
|
|
|
|
if (pts == MP_NOPTS_VALUE) {
|
|
|
|
mp_msg(MSGT_SUBREADER, MSGL_WARN, "Subtitle without pts, ignored\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
long long ipts = pts * 1000 + 0.5;
|
|
|
|
long long iduration = duration * 1000 + 0.5;
|
|
|
|
if (ctx->incomplete_event) {
|
|
|
|
ctx->incomplete_event = false;
|
|
|
|
ASS_Event *event = track->events + track->n_events - 1;
|
|
|
|
if (ipts <= event->Start)
|
|
|
|
free_last_event(track);
|
|
|
|
else
|
|
|
|
event->Duration = ipts - event->Start;
|
|
|
|
}
|
|
|
|
// Note: we rely on there being guaranteed 0 bytes after data packets
|
|
|
|
int len = strlen(text);
|
|
|
|
if (len < 5) {
|
|
|
|
// Some tracks use a whitespace (but not empty) packet to mark end
|
|
|
|
// of previous subtitle.
|
|
|
|
for (int i = 0; i < len; i++)
|
|
|
|
if (!strchr(" \f\n\r\t\v", text[i]))
|
|
|
|
goto not_all_whitespace;
|
|
|
|
return;
|
2011-01-16 18:03:08 +00:00
|
|
|
}
|
2011-01-18 13:33:36 +00:00
|
|
|
not_all_whitespace:;
|
|
|
|
char buf[500];
|
|
|
|
subassconvert_subrip(text, buf, sizeof(buf));
|
|
|
|
for (int i = 0; i < track->n_events; i++)
|
|
|
|
if (track->events[i].Start == ipts
|
|
|
|
&& (duration <= 0 || track->events[i].Duration == iduration)
|
|
|
|
&& strcmp(track->events[i].Text, buf) == 0)
|
|
|
|
return; // We've already added this subtitle
|
|
|
|
if (duration <= 0) {
|
|
|
|
iduration = 10000;
|
|
|
|
ctx->incomplete_event = true;
|
|
|
|
}
|
|
|
|
int eid = ass_alloc_event(track);
|
|
|
|
ASS_Event *event = track->events + eid;
|
|
|
|
event->Start = ipts;
|
|
|
|
event->Duration = iduration;
|
|
|
|
event->Text = strdup(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void reset(struct sh_sub *sh, struct osd_state *osd)
|
|
|
|
{
|
|
|
|
struct sd_ass_priv *ctx = sh->context;
|
|
|
|
if (ctx->incomplete_event)
|
|
|
|
free_last_event(ctx->ass_track);
|
|
|
|
ctx->incomplete_event = false;
|
2011-01-16 18:03:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void switch_off(struct sh_sub *sh, struct osd_state *osd)
|
|
|
|
{
|
2011-01-18 13:33:36 +00:00
|
|
|
reset(sh, osd);
|
2011-01-16 18:03:08 +00:00
|
|
|
osd->ass_track = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void uninit(struct sh_sub *sh)
|
|
|
|
{
|
|
|
|
struct sd_ass_priv *ctx = sh->context;
|
|
|
|
|
|
|
|
ass_free_track(ctx->ass_track);
|
|
|
|
talloc_free(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct sd_functions sd_ass = {
|
|
|
|
.init = init,
|
|
|
|
.decode = decode,
|
2011-01-18 13:33:36 +00:00
|
|
|
.reset = reset,
|
2011-01-16 18:03:08 +00:00
|
|
|
.switch_off = switch_off,
|
|
|
|
.uninit = uninit,
|
|
|
|
};
|