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.
113 lines
2.7 KiB
113 lines
2.7 KiB
package Subtitle::MSub; |
|
|
|
use strict; |
|
use warnings; |
|
use utf8; |
|
|
|
use Subtitle::Utils qw(:string); |
|
|
|
use base 'Subtitle::BASE'; |
|
|
|
sub new { |
|
my ($class, %args) = @_; |
|
my $self = { |
|
debug => 0, |
|
eol => "\n", |
|
fps => undef, |
|
default_fps => 25.0, |
|
convert_timing => 1, |
|
%args, |
|
events => [], |
|
log => [], |
|
}; |
|
|
|
return bless($self, $class); |
|
} |
|
|
|
sub new_event { return +{ timing => undef, text => undef }; } |
|
|
|
sub parse { |
|
my ($self, $lines) = @_; |
|
my $linenum = 0; |
|
my $event; |
|
|
|
foreach my $line (@{ $lines }) { |
|
$linenum++; |
|
chomp_all($line); |
|
trim($line); |
|
next unless $line; |
|
my ($start, $end, $rest) = ($line =~ m/{(\d+)}\s*{(\d+)}\s*(.+)/o); |
|
# expected: garbage |
|
unless ($start and $end and $rest) { |
|
$self->log(warn => "Unrecognized line at $linenum: $line"); |
|
next; |
|
} |
|
# expected: valid line |
|
# special case - '{1}{1}XX.YY' line sets fps |
|
if ($start == 1 and $end == 1) { |
|
my ($fps) = ($rest =~ m/(\d+(?:\.\d+))/o); |
|
unless ($fps) { |
|
$self->log(error => "Expected fps at line $linenum, but found: $rest"); |
|
next; |
|
} |
|
if ($fps and $self->{fps}) { |
|
$self->log(warn => "Found fps line at $linenum, but fps already set by user"); |
|
next; |
|
} |
|
$self->log(debug => "Set fps to $fps and line $linenum"); |
|
$self->{fps} = $fps; |
|
next; |
|
} |
|
trim($rest); |
|
$rest =~ s/\|/$self->{eol}/og; |
|
my $event = $self->new_event; |
|
$event->{timing} = [$start, $end]; |
|
$event->{text} = $rest; |
|
push @{ $self->{events} }, $event; |
|
undef $event; |
|
} |
|
# set fps if none and recalc timing |
|
$self->{fps} //= $self->{default_fps}; |
|
return scalar @{ $self->{events} } |
|
unless $self->{convert_timing}; |
|
|
|
$self->log(debug => "Converting frame numbers to time with fps $self->{fps}"); |
|
foreach my $event (@{ $self->{events} }) { |
|
$event->{timing}->[0] /= $self->{fps}; |
|
$event->{timing}->[1] /= $self->{fps}; |
|
} |
|
|
|
return scalar @{ $self->{events} }; |
|
} |
|
|
|
1; |
|
|
|
=pod |
|
|
|
=head1 NAME |
|
|
|
Subtitle::MSub -- micsrosub subtitle format |
|
|
|
=head1 METHODS |
|
|
|
=head2 C<new> |
|
|
|
my $msub = Subtitle::MSub->new(%opts); |
|
|
|
Creates new object, takes list of options as argument. |
|
Recognized keys are: |
|
|
|
* debug -- write debug messages to log (default: 0) |
|
* eol -- replace "|" symbol within subtitle text with this chars (default: "\n") |
|
* fps -- sets FPS for subtitle (default: not set, try detect from file, then use default) |
|
* default_fps -- FPS for subtitle, if not found in file, and not set by user (default: 25.0) |
|
* convert_timing -- convert frame numbers to seconds (default: 1) |
|
|
|
=head2 C<parse> |
|
|
|
my $events_count = $msub->parse(\@lines); |
|
|
|
Parse subtitle from array of lines. |
|
Returns parsed events count on success, -1 on error |
|
|
|
=cut
|
|
|