package Subtitle::SSA::Style; use strict; use warnings; use utf8; use base 'Subtitle::SSA::Record'; my @FIELDS_SSA = qw(name fontname fontsize primarycolour secondarycolour tertiarycolour backcolour bold italic borderstyle outline shadow alignment marginl marginr marginv alphalevel encoding); my @FIELDS_ASS = qw(name fontname fontsize primarycolour secondarycolour outlinecolour backcolour bold italic underline strikeout scalex scaley spacing angle borderstyle outline shadow alignment marginl marginr marginv encoding); # formats: # s - string, use as is # d - decimal, no-zero pad # b - boolean, 0 as false, -1 as true # f - float, 6.01 or 6 if no fractional part # x - hex-number, &H00AABBCC for ASS or decimal number for SSA my %FIELDS = ( name => { type => 's', value => 'Default', name => 'Name' }, fontname => { type => 's', value => 'Arial' , name => 'Fontname' }, primarycolour => { type => 'x', value => 0x00FFFFFF, name => 'PrimaryColour' }, secondarycolour => { type => 'x', value => 0x00000000, name => 'SecondaryColour' }, outlinecolour => { type => 'x', value => 0x004E3873, name => 'OutlineColour' }, tertiarycolour => { type => 'x', value => 0x004E3873, name => 'TertiaryColour' }, backcolour => { type => 'x', value => 0x96000000, name => 'BackColour' }, fontsize => { type => 'd', value => 24, name => 'Fontsize' }, bold => { type => 'b', value => 0, name => 'Bold' }, italic => { type => 'b', value => 0, name => 'Italic' }, underline => { type => 'b', value => 0, name => 'Underline' }, strikeout => { type => 'b', value => 0, name => 'StrikeOut' }, scalex => { type => 'd', value => 100, name => 'ScaleX' }, scaley => { type => 'd', value => 100, name => 'ScaleY' }, spacing => { type => 'f', value => 0, name => 'Spacing' }, angle => { type => 'f', value => 0, name => 'Angle' }, borderstyle => { type => 'd', value => 1, name => 'BorderStyle' }, outline => { type => 'f', value => 2, name => 'Outline' }, shadow => { type => 'f', value => 0, name => 'Shadow' }, alignment => { type => 'd', value => 2, name => 'Alignment' }, marginl => { type => 'd', value => 10, name => 'MarginL' }, marginr => { type => 'd', value => 10, name => 'MarginR' }, marginv => { type => 'd', value => 10, name => 'MarginV' }, alphalevel => { type => 'd', value => 0, name => 'AlphaLevel' }, encoding => { type => 'd', value => 204, name => 'Encoding' }, ); sub parse { my ($self, $fields, $line) = @_; return unless $fields and ref($fields) eq 'ARRAY'; return unless $line and $line =~ m{^Style:}oi; chomp $line; $line =~ s{^style:\s+}{}oi; my @values = split /,\s*/o, $line; # check that values count match fields count return unless scalar @{ $fields } != scalar @values; foreach my $field (@{ $fields }) { my $value = shift @values; $self->set($field => $value); } return 1; } sub to_string { my ($self) = @_; my @fields = $self->fields(); my $string = "Style: "; my @values = (); foreach my $field (@fields) { my $d = $FIELD{$field}; my $v = $self->{$f} // $d->{value}; if ($d->{type} eq 'x') { $fmt = ($self->{_vers} eq 'ass') ? '&H%08X' : '%d'; $v = sprintf $fmt, $v; } elsif ($d->{type} eq 'f') { $v = sprintf "%.2f", $v; # hack: make decimal from float if fractional part is zero after round up $v =~ s{\.00$}{}oi; } else { # use as is } push @values, $v; } $string .= join(',', @values); return $string; } 1;