X-Git-Url: http://git.shiar.nl/sheet.git/blobdiff_plain/de64283eddcfa5e620386896983841f47b1de8d4..8f02cfee34ce655998ce67264e35a091c7d5491d:/sc.plp diff --git a/sc.plp b/sc.plp index 45a9f7b..2217dc6 100644 --- a/sc.plp +++ b/sc.plp @@ -1,4 +1,5 @@ <(common.inc.plp)><: +use List::Util qw(max); my %scver = ( id => 'bw', @@ -46,29 +47,29 @@ die "Cannot open unit data: $_\n" for $@ || $! || (); my $patch = shift @{$units} or die "Cannot open unit data: metadata not found\n"; -print "

Unit properties as seen or measured in $scver{name}\n$patch.\n

\n\n"; +print "

Unit properties as seen or measured in $scver{name}\n$patch.\n"; +print "Also see the $_ table.\n" for join(', ', + ('StarCraft 2: HotS') x ($scver{major} < 2), + ('original SC: Brood War') x ($scver{major} > 1), +); +print "

\n\n"; + +sub addupgrade { + my ($ref, $increase) = @_; + if (ref $increase eq 'HASH') { + addupgrade(\${$ref}->{$_}, $increase->{$_}) for keys %{$increase}; + } + elsif (ref $increase eq 'ARRAY') { + addupgrade(\${$ref}->[$_], $increase->[$_]) for 0 .. $#{$increase}; + } + ${$ref} += $increase if $increase =~ /^-?[0-9.]+/; +} for my $unit (@{$units}) { - for my $upgrade (@{ $unit->{upgrade} // $unit->{update} // [] }) { + for my $upgrade (@{ $unit->{upgrade} }) { while (my ($col, $increase) = each %{$upgrade}) { defined $unit->{$col} or next; - - if (ref $increase eq 'ARRAY') { - for my $incrow (0 .. $#{$increase}) { - while (my ($inccol, $subinc) = each %{ $increase->[$incrow] }) { - $unit->{upgraded}->{$col}->[$incrow]->{$inccol} - //= ref $unit->{$col}->[$incrow]->{$inccol} - ? $unit->{$col}->[$incrow]->{$inccol}->[-1] - : $unit->{$col}->[$incrow]->{$inccol}; - $unit->{upgraded}->{$col}->[$incrow]->{$inccol} - += ref $subinc ? $subinc->[-1] : $subinc; - } - } - } - elsif ($increase =~ /^-?[0-9.]+/) { - $unit->{upgraded}->{$col} //= $unit->{$col}; - $unit->{upgraded}->{$col} += $increase; - } + addupgrade(\( $unit->{upgraded}->{$col} //= $unit->{$col} ), $increase); } } } @@ -102,43 +103,10 @@ sub coltoggle { <: sub showrange { my ($min, $max) = @_; - $_ = int($_ + .5) for $min, $max; # round halves up - return $min == $max ? $min : "$min-$max"; -} - -sub showval { - my ($row, @elements) = @_; - my ($min, $max); - - my $value = $row; - $value = ref $value eq 'HASH' ? $value->{$_} - : ref $value eq 'ARRAY' && $value->[$_] or last for @elements; - if (ref $value eq 'ARRAY') { - $min = $value->[0]; - $max = $value->[-1]; - } - else { - $min = $max = $value; - } - defined $min or return ''; - - if (my $increase = $row->{upgraded}) { - $increase = ref $increase eq 'HASH' ? $increase->{$_} - : ref $increase eq 'ARRAY' && $increase->[$_] or last for @elements; - $max = $increase if $increase; - } - - if ($elements[0] eq 'attack' and $elements[2] eq 'dps') {{ - my $attack = $row->{ $elements[0] }->[ $elements[1] ]; - my $type = $attack->{type} or next; - if ($type eq 'explosive') { - $min /= 2; - } - elsif ($type eq 'implosive') { - $min /= 4; - } - }} - return showrange($min, $max); + return '' if not defined $min; + $_ &&= int($_ + .5) for $min, $max; # round halves up + return $min if not defined $max or $min == $max; + return "$min-$max"; } sub showattack { @@ -146,32 +114,67 @@ sub showval { my $attack = $row->{attack}->[$area] or return ''; - my $tagbase = ''; - $tagbase .= '' . '▽' x !!($attack->{anti} & 1); - $tagbase .= '' . '△' x !!($attack->{anti} & 2); - $tagbase .= ''; + my $out = ''; + $out .= '' . '▽' x !!($attack->{anti} & 1); + $out .= '' . '△' x !!($attack->{anti} & 2); + $out .= ''; + + my $upattack = $row->{upgraded}->{attack}->[$area]; + my $damage = $attack->{damage}; + my $maxdamage = $upattack->{damage} // $damage; + $damage = $damage->[0] if ref $damage; + $maxdamage = $maxdamage->[-1] if ref $maxdamage; - my $out = showval($row, 'attack', $area, 'damage'); + $out .= "$attack->{count}× " if $attack->{count} > 1; + $out .= '*' + if $attack->{type} eq 'explosive'; + $out .= '~' + if $attack->{type} eq 'implosive'; + $out .= sprintf('', + (map { + $_ =~ /^light/ ? 'unit-s' : + $_ eq 'armored' ? 'unit-l' : + $_ eq 'organic' ? 'unit-o' : + $_ =~ /^massive/ ? 'unit-h' : + $_ eq 'shields' ? 'unit-shield' : + '', + } join '_', keys %{ $attack->{bonus} }), + join(', ', map {( + sprintf('+%s vs %s', + (map { + ref $_ ? showrange($_->[0], $_->[-1]) : $_ + } $attack->{bonus}->{$_}), + $_, + ), + )} keys %{ $attack->{bonus} }), + ) if $attack->{bonus}; + $out .= showrange($damage, $maxdamage); $out .= sprintf('%s', $attack->{splash} eq 'line' ? ('linear', '×') : ('splash', '+') ) if $attack->{splash}; - $attack->{dps} = $attack->{cooldown} && [ - map { $_ / $attack->{cooldown} * ($attack->{count} // 1) } - map { ref $_ ? @{$_} : $_ } - $attack->{damage} - #TODO: upgrade (zergling) - ]; - $out .= '' . showval($row, 'attack', $area, 'dps'); - return $tagbase . $out; + + $out .= ''; + if ($attack->{cooldown}) { + if (my $type = $attack->{type}) { + if ($type eq 'explosive') { + $damage /= 2; + } + elsif ($type eq 'implosive') { + $damage /= 4; + } + } + $damage *= ($attack->{count} // 1) / $attack->{cooldown}; + if (my $bonus = $upattack->{bonus} // $attack->{bonus}) { + $maxdamage += $_ for max( + map { ref $_ ? $_->[-1] : $_ } values %{$bonus} + ); + } + $maxdamage *= ($upattack->{count} // $attack->{count} // 1) + / ($upattack->{cooldown} // $attack->{cooldown}); + $out .= showrange($damage, $maxdamage); + } + + return $out; } sub showmagic { @@ -254,14 +257,17 @@ sub showval { $_->{shield} ? sprintf('%.0f%%{shield} / $_->{hp} ) : '' . showval($_, 'armor'), + ' class="val unit-armor">' . + showrange($_->{armor}, $_->{upgraded}->{armor}), showattack($_, 0), - '' . showval($_, 'attack', 0, 'range'), + '' . + showrange(map { $_->{attack}->[0]->{range} } $_, $_->{upgraded}), '' . sprintf( $_->{detect} ? '%s' : '%s', - showval($_, 'sight') + showrange($_->{sight}, $_->{upgraded}->{sight}) ), - '' . showval($_, 'speed'), + '' . + showrange($_->{speed}, $_->{upgraded}->{speed}), $_->{attr}->{jump} && qq'↕', $_->{attr}->{flying} @@ -279,10 +285,15 @@ sub showval { $grouped = 0; $get{order} ||= ''; if ($get{order} eq 'size') { - $_->{order} = $_->{unit}*8 + $_->{suit} + $_->{hp}/512 + $_->{min}/8192 for @$units; + $_->{order} = ( + $_->{unit}*16 + ($_->{size} // $_->{suit}) + $_->{cargo}/8 + + $_->{hp}/512 + $_->{min}/8192 + ) for @$units; } elsif ($get{order} eq 'cost') { - $_->{order} = $_->{gas}*1.5 + $_->{min} + $_->{unit}/8 + $_->{build}/256/8 for @$units; + $_->{order} = ( + $_->{gas}*1.5 + $_->{min} + $_->{unit}/8 + $_->{build}/256/8 + ) for @$units; } else { $units->[$_]->{order} = $_ for 0 .. $#$units; @@ -374,14 +385,20 @@ if ($scver{major} > 1) {
targets ▽ ground and/or △ air
damage given per single hit -
dps indicates relative amount of damage done in - 1 second of in-game time +
dps indicates relative amount of damage + done in 1 second of in-game time
splash damage hits all objects nearby + or in a straight line ×. -
explosive damage does only +
<: +if ($scver{major} > 1) { + :>does not include bonus damage + dealt to susceptible unit types<: +} else { + :>*explosive damage does only 50% damage to small units, 75% to medium, 100% to large -
concussive/plasma damage does - 25% to large, 50% medium, 100% to small units +
~concussive/plasma damage does + 25% to large, 50% medium, 100% to small units<: +} :>
sight
range in which the unit detects other units
emphasis indicates ability to detect cloaked units