1081 lines
29 KiB
Perl
1081 lines
29 KiB
Perl
#================================================================= -*-Perl-*-
|
|
#
|
|
# Template::Directive
|
|
#
|
|
# DESCRIPTION
|
|
# Factory module for constructing templates from Perl code.
|
|
#
|
|
# AUTHOR
|
|
# Andy Wardley <abw@wardley.org>
|
|
#
|
|
# WARNING
|
|
# Much of this module is hairy, even furry in places. It needs
|
|
# a lot of tidying up and may even be moved into a different place
|
|
# altogether. The generator code is often inefficient, particularly in
|
|
# being very anal about pretty-printing the Perl code all neatly, but
|
|
# at the moment, that's still high priority for the sake of easier
|
|
# debugging.
|
|
#
|
|
# COPYRIGHT
|
|
# Copyright (C) 1996-2007 Andy Wardley. All Rights Reserved.
|
|
#
|
|
# This module is free software; you can redistribute it and/or
|
|
# modify it under the same terms as Perl itself.
|
|
#
|
|
#============================================================================
|
|
|
|
package Template::Directive;
|
|
|
|
use strict;
|
|
use warnings;
|
|
use base 'Template::Base';
|
|
use Template::Constants;
|
|
use Template::Exception;
|
|
|
|
our $VERSION = '3.009';
|
|
our $DEBUG = 0 unless defined $DEBUG;
|
|
our $WHILE_MAX = 1000 unless defined $WHILE_MAX;
|
|
our $PRETTY = 0 unless defined $PRETTY;
|
|
our $OUTPUT = '$output .= ';
|
|
|
|
|
|
sub _init {
|
|
my ($self, $config) = @_;
|
|
$self->{ NAMESPACE } = $config->{ NAMESPACE };
|
|
return $self;
|
|
}
|
|
|
|
sub trace_vars {
|
|
my $self = shift;
|
|
return @_
|
|
? ($self->{ TRACE_VARS } = shift)
|
|
: $self->{ TRACE_VARS };
|
|
}
|
|
|
|
sub pad {
|
|
my ($text, $pad) = @_;
|
|
$pad = ' ' x ($pad * 4);
|
|
$text =~ s/^(?!#line)/$pad/gm;
|
|
$text;
|
|
}
|
|
|
|
#========================================================================
|
|
# FACTORY METHODS
|
|
#
|
|
# These methods are called by the parser to construct directive instances.
|
|
#========================================================================
|
|
|
|
#------------------------------------------------------------------------
|
|
# template($block)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub template {
|
|
my ($self, $block) = @_;
|
|
$block = pad($block, 2) if $PRETTY;
|
|
|
|
return "sub { return '' }" unless $block =~ /\S/;
|
|
|
|
return <<EOF;
|
|
sub {
|
|
my \$context = shift || die "template sub called without context\\n";
|
|
my \$stash = \$context->stash;
|
|
my \$output = '';
|
|
my \$_tt_error;
|
|
|
|
eval { BLOCK: {
|
|
$block
|
|
} };
|
|
if (\$@) {
|
|
\$_tt_error = \$context->catch(\$@, \\\$output);
|
|
die \$_tt_error unless \$_tt_error->type eq 'return';
|
|
}
|
|
|
|
return \$output;
|
|
}
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# anon_block($block) [% BLOCK %] ... [% END %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub anon_block {
|
|
my ($self, $block) = @_;
|
|
$block = pad($block, 2) if $PRETTY;
|
|
|
|
return <<EOF;
|
|
|
|
# BLOCK
|
|
$OUTPUT do {
|
|
my \$output = '';
|
|
my \$_tt_error;
|
|
|
|
eval { BLOCK: {
|
|
$block
|
|
} };
|
|
if (\$@) {
|
|
\$_tt_error = \$context->catch(\$@, \\\$output);
|
|
die \$_tt_error unless \$_tt_error->type eq 'return';
|
|
}
|
|
|
|
\$output;
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# block($blocktext)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub block {
|
|
my ($self, $block) = @_;
|
|
return join("\n", @{ $block || [] });
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# textblock($text)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub textblock {
|
|
my ($self, $text) = @_;
|
|
return "$OUTPUT " . &text($self, $text) . ';';
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# text($text)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub text {
|
|
my ( $self, $text ) = @_;
|
|
|
|
return '' if !length $text;
|
|
|
|
if ( $text =~ tr{$@\\}{} ) {
|
|
$text =~ s/(["\$\@\\])/\\$1/g;
|
|
$text =~ s/\n/\\n/g;
|
|
return '"' . $text . '"';
|
|
}
|
|
|
|
$text =~ s{'}{\\'}g if index( $text, q{'} ) != -1;
|
|
return q{'} . $text . q{'};
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# quoted(\@items) "foo$bar"
|
|
#------------------------------------------------------------------------
|
|
|
|
sub quoted {
|
|
my ($self, $items) = @_;
|
|
return '' unless @$items;
|
|
return ("('' . " . $items->[0] . ')') if scalar @$items == 1;
|
|
return '(' . join(' . ', @$items) . ')';
|
|
# my $r = '(' . join(' . ', @$items) . ' . "")';
|
|
# print STDERR "[$r]\n";
|
|
# return $r;
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# ident(\@ident) foo.bar(baz)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub ident {
|
|
my ($self, $ident) = @_;
|
|
return "''" unless @$ident;
|
|
my $ns;
|
|
|
|
# Careful! Template::Parser always creates a Template::Directive object
|
|
# (as of v2.22_1) so $self is usually an object. However, we used to
|
|
# allow Template::Directive methods to be called as class methods and
|
|
# Template::Namespace::Constants module takes advantage of this fact
|
|
# by calling Template::Directive->ident() when it needs to generate an
|
|
# identifier. This hack guards against Mr Fuckup from coming to town
|
|
# when that happens.
|
|
|
|
if (ref $self) {
|
|
# trace variable usage
|
|
if ($self->{ TRACE_VARS }) {
|
|
my $root = $self->{ TRACE_VARS };
|
|
my $n = 0;
|
|
my $v;
|
|
while ($n < @$ident) {
|
|
$v = $ident->[$n];
|
|
for ($v) { s/^'//; s/'$// };
|
|
$root = $root->{ $v } ||= { };
|
|
$n += 2;
|
|
}
|
|
}
|
|
|
|
# does the first element of the identifier have a NAMESPACE
|
|
# handler defined?
|
|
if (@$ident > 2 && ($ns = $self->{ NAMESPACE })) {
|
|
my $key = $ident->[0];
|
|
|
|
# a faster alternate to $key =~ s/^'(.+)'$/$1/s
|
|
if ( index( $key, q[']) == 0 ) {
|
|
substr( $key, 0, 1, '' );
|
|
substr( $key, -1, 1, '' ); # remove the last char blindly
|
|
}
|
|
|
|
if ($ns = $ns->{ $key }) {
|
|
return $ns->ident($ident);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (scalar @$ident <= 2 && ! $ident->[1]) {
|
|
$ident = $ident->[0];
|
|
}
|
|
else {
|
|
$ident = '[' . join(', ', @$ident) . ']';
|
|
}
|
|
return "\$stash->get($ident)";
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# identref(\@ident) \foo.bar(baz)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub identref {
|
|
my ($self, $ident) = @_;
|
|
return "''" unless @$ident;
|
|
if (scalar @$ident <= 2 && ! $ident->[1]) {
|
|
$ident = $ident->[0];
|
|
}
|
|
else {
|
|
$ident = '[' . join(', ', @$ident) . ']';
|
|
}
|
|
return "\$stash->getref($ident)";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# assign(\@ident, $value, $default) foo = bar
|
|
#------------------------------------------------------------------------
|
|
|
|
sub assign {
|
|
my ($self, $var, $val, $default) = @_;
|
|
|
|
if (ref $var) {
|
|
if (scalar @$var == 2 && ! $var->[1]) {
|
|
$var = $var->[0];
|
|
}
|
|
else {
|
|
$var = '[' . join(', ', @$var) . ']';
|
|
}
|
|
}
|
|
$val .= ', 1' if $default;
|
|
return "\$stash->set($var, $val)";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# args(\@args) foo, bar, baz = qux
|
|
#------------------------------------------------------------------------
|
|
|
|
sub args {
|
|
my ($self, $args) = @_;
|
|
my $hash = shift @$args;
|
|
push(@$args, '{ ' . join(', ', @$hash) . ' }')
|
|
if @$hash;
|
|
|
|
return '0' unless @$args;
|
|
return '[ ' . join(', ', @$args) . ' ]';
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# filenames(\@names)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub filenames {
|
|
my ($self, $names) = @_;
|
|
if (@$names > 1) {
|
|
$names = '[ ' . join(', ', @$names) . ' ]';
|
|
}
|
|
else {
|
|
$names = shift @$names;
|
|
}
|
|
return $names;
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# get($expr) [% foo %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub get {
|
|
my ($self, $expr) = @_;
|
|
return "$OUTPUT $expr;";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# call($expr) [% CALL bar %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub call {
|
|
my ($self, $expr) = @_;
|
|
$expr .= ';';
|
|
return $expr;
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# set(\@setlist) [% foo = bar, baz = qux %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub set {
|
|
my ($self, $setlist) = @_;
|
|
my $output;
|
|
while (my ($var, $val) = splice(@$setlist, 0, 2)) {
|
|
$output .= &assign($self, $var, $val) . ";\n";
|
|
}
|
|
chomp $output;
|
|
return $output;
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# default(\@setlist) [% DEFAULT foo = bar, baz = qux %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub default {
|
|
my ($self, $setlist) = @_;
|
|
my $output;
|
|
while (my ($var, $val) = splice(@$setlist, 0, 2)) {
|
|
$output .= &assign($self, $var, $val, 1) . ";\n";
|
|
}
|
|
chomp $output;
|
|
return $output;
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# insert(\@nameargs) [% INSERT file %]
|
|
# # => [ [ $file, ... ], \@args ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub insert {
|
|
my ($self, $nameargs) = @_;
|
|
my ($file, $args) = @$nameargs;
|
|
$file = $self->filenames($file);
|
|
return "$OUTPUT \$context->insert($file);";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# include(\@nameargs) [% INCLUDE template foo = bar %]
|
|
# # => [ [ $file, ... ], \@args ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub include {
|
|
my ($self, $nameargs) = @_;
|
|
my ($file, $args) = @$nameargs;
|
|
my $hash = shift @$args;
|
|
$file = $self->filenames($file);
|
|
$file .= @$hash ? ', { ' . join(', ', @$hash) . ' }' : '';
|
|
return "$OUTPUT \$context->include($file);";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# process(\@nameargs) [% PROCESS template foo = bar %]
|
|
# # => [ [ $file, ... ], \@args ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub process {
|
|
my ($self, $nameargs) = @_;
|
|
my ($file, $args) = @$nameargs;
|
|
my $hash = shift @$args;
|
|
$file = $self->filenames($file);
|
|
$file .= @$hash ? ', { ' . join(', ', @$hash) . ' }' : '';
|
|
return "$OUTPUT \$context->process($file);";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# if($expr, $block, $else) [% IF foo < bar %]
|
|
# ...
|
|
# [% ELSE %]
|
|
# ...
|
|
# [% END %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub if {
|
|
my ($self, $expr, $block, $else) = @_;
|
|
my @else = $else ? @$else : ();
|
|
$else = pop @else;
|
|
$block = pad($block, 1) if $PRETTY;
|
|
|
|
my $output = "if ($expr) {\n$block\n}\n";
|
|
|
|
foreach my $elsif (@else) {
|
|
($expr, $block) = @$elsif;
|
|
$block = pad($block, 1) if $PRETTY;
|
|
$output .= "elsif ($expr) {\n$block\n}\n";
|
|
}
|
|
if (defined $else) {
|
|
$else = pad($else, 1) if $PRETTY;
|
|
$output .= "else {\n$else\n}\n";
|
|
}
|
|
|
|
return $output;
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# foreach($target, $list, $args, $block) [% FOREACH x = [ foo bar ] %]
|
|
# ...
|
|
# [% END %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub foreach {
|
|
my ($self, $target, $list, $args, $block, $label) = @_;
|
|
$args = shift @$args;
|
|
$args = @$args ? ', { ' . join(', ', @$args) . ' }' : '';
|
|
$label ||= 'LOOP';
|
|
|
|
my ($loop_save, $loop_set, $loop_restore, $setiter);
|
|
if ($target) {
|
|
$loop_save = 'eval { $_tt_oldloop = ' . &ident($self, ["'loop'"]) . ' }';
|
|
$loop_set = "\$stash->{'$target'} = \$_tt_value";
|
|
$loop_restore = "\$stash->set('loop', \$_tt_oldloop)";
|
|
}
|
|
else {
|
|
$loop_save = '$stash = $context->localise()';
|
|
# $loop_set = "\$stash->set('import', \$_tt_value) "
|
|
# . "if ref \$value eq 'HASH'";
|
|
$loop_set = "\$stash->get(['import', [\$_tt_value]]) "
|
|
. "if ref \$_tt_value eq 'HASH'";
|
|
$loop_restore = '$stash = $context->delocalise()';
|
|
}
|
|
$block = pad($block, 3) if $PRETTY;
|
|
|
|
return <<EOF;
|
|
|
|
# FOREACH
|
|
do {
|
|
my (\$_tt_value, \$_tt_error, \$_tt_oldloop);
|
|
my \$_tt_list = $list;
|
|
|
|
unless (UNIVERSAL::isa(\$_tt_list, 'Template::Iterator')) {
|
|
\$_tt_list = Template::Config->iterator(\$_tt_list)
|
|
|| die \$Template::Config::ERROR, "\\n";
|
|
}
|
|
|
|
(\$_tt_value, \$_tt_error) = \$_tt_list->get_first();
|
|
$loop_save;
|
|
\$stash->set('loop', \$_tt_list);
|
|
eval {
|
|
$label: while (! \$_tt_error) {
|
|
$loop_set;
|
|
$block;
|
|
(\$_tt_value, \$_tt_error) = \$_tt_list->get_next();
|
|
}
|
|
};
|
|
$loop_restore;
|
|
die \$@ if \$@;
|
|
\$_tt_error = 0 if \$_tt_error && \$_tt_error eq Template::Constants::STATUS_DONE;
|
|
die \$_tt_error if \$_tt_error;
|
|
};
|
|
EOF
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# next() [% NEXT %]
|
|
#
|
|
# Next iteration of a FOREACH loop (experimental)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub next {
|
|
my ($self, $label) = @_;
|
|
$label ||= 'LOOP';
|
|
return <<EOF;
|
|
(\$_tt_value, \$_tt_error) = \$_tt_list->get_next();
|
|
next $label;
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# wrapper(\@nameargs, $block) [% WRAPPER template foo = bar %]
|
|
# # => [ [$file,...], \@args ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub wrapper {
|
|
my ($self, $nameargs, $block) = @_;
|
|
my ($file, $args) = @$nameargs;
|
|
my $hash = shift @$args;
|
|
|
|
local $" = ', ';
|
|
# print STDERR "wrapper([@$file], { @$hash })\n";
|
|
|
|
return $self->multi_wrapper($file, $hash, $block)
|
|
if @$file > 1;
|
|
$file = shift @$file;
|
|
|
|
$block = pad($block, 1) if $PRETTY;
|
|
push(@$hash, "'content'", '$output');
|
|
$file .= @$hash ? ', { ' . join(', ', @$hash) . ' }' : '';
|
|
|
|
return <<EOF;
|
|
|
|
# WRAPPER
|
|
$OUTPUT do {
|
|
my \$output = '';
|
|
$block
|
|
\$context->include($file);
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
sub multi_wrapper {
|
|
my ($self, $file, $hash, $block) = @_;
|
|
$block = pad($block, 1) if $PRETTY;
|
|
|
|
push(@$hash, "'content'", '$output');
|
|
$hash = @$hash ? ', { ' . join(', ', @$hash) . ' }' : '';
|
|
|
|
$file = join(', ', reverse @$file);
|
|
# print STDERR "multi wrapper: $file\n";
|
|
|
|
return <<EOF;
|
|
|
|
# WRAPPER
|
|
$OUTPUT do {
|
|
my \$output = '';
|
|
$block
|
|
foreach ($file) {
|
|
\$output = \$context->include(\$_$hash);
|
|
}
|
|
\$output;
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# while($expr, $block) [% WHILE x < 10 %]
|
|
# ...
|
|
# [% END %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub while {
|
|
my ($self, $expr, $block, $label) = @_;
|
|
$block = pad($block, 2) if $PRETTY;
|
|
$label ||= 'LOOP';
|
|
|
|
return <<EOF;
|
|
|
|
# WHILE
|
|
do {
|
|
my \$_tt_failsafe = $WHILE_MAX;
|
|
$label:
|
|
while (($expr) && --\$_tt_failsafe >= 0) {
|
|
$block
|
|
}
|
|
die "WHILE loop terminated (> $WHILE_MAX iterations)\\n"
|
|
if \$_tt_failsafe < 0;
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# switch($expr, \@case) [% SWITCH %]
|
|
# [% CASE foo %]
|
|
# ...
|
|
# [% END %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub switch {
|
|
my ($self, $expr, $case) = @_;
|
|
my @case = @$case;
|
|
my ($match, $block, $default);
|
|
my $caseblock = '';
|
|
|
|
$default = pop @case;
|
|
|
|
foreach $case (@case) {
|
|
$match = $case->[0];
|
|
$block = $case->[1];
|
|
$block = pad($block, 1) if $PRETTY;
|
|
$caseblock .= <<EOF;
|
|
\$_tt_match = $match;
|
|
\$_tt_match = [ \$_tt_match ] unless ref \$_tt_match eq 'ARRAY';
|
|
if (grep(/^\\Q\$_tt_result\\E\$/, \@\$_tt_match)) {
|
|
$block
|
|
last SWITCH;
|
|
}
|
|
EOF
|
|
}
|
|
|
|
$caseblock .= $default
|
|
if defined $default;
|
|
$caseblock = pad($caseblock, 2) if $PRETTY;
|
|
|
|
return <<EOF;
|
|
|
|
# SWITCH
|
|
do {
|
|
my \$_tt_result = $expr;
|
|
my \$_tt_match;
|
|
SWITCH: {
|
|
$caseblock
|
|
}
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# try($block, \@catch) [% TRY %]
|
|
# ...
|
|
# [% CATCH %]
|
|
# ...
|
|
# [% END %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub try {
|
|
my ($self, $block, $catch) = @_;
|
|
my @catch = @$catch;
|
|
my ($match, $mblock, $default, $final, $n);
|
|
my $catchblock = '';
|
|
my $handlers = [];
|
|
|
|
$block = pad($block, 2) if $PRETTY;
|
|
$final = pop @catch;
|
|
$final = "# FINAL\n" . ($final ? "$final\n" : '')
|
|
. 'die $_tt_error if $_tt_error;' . "\n" . '$output;';
|
|
$final = pad($final, 1) if $PRETTY;
|
|
|
|
$n = 0;
|
|
foreach $catch (@catch) {
|
|
$match = $catch->[0] || do {
|
|
$default ||= $catch->[1];
|
|
next;
|
|
};
|
|
$mblock = $catch->[1];
|
|
$mblock = pad($mblock, 1) if $PRETTY;
|
|
push(@$handlers, "'$match'");
|
|
$catchblock .= $n++
|
|
? "elsif (\$_tt_handler eq '$match') {\n$mblock\n}\n"
|
|
: "if (\$_tt_handler eq '$match') {\n$mblock\n}\n";
|
|
}
|
|
$catchblock .= "\$_tt_error = 0;";
|
|
$catchblock = pad($catchblock, 3) if $PRETTY;
|
|
if ($default) {
|
|
$default = pad($default, 1) if $PRETTY;
|
|
$default = "else {\n # DEFAULT\n$default\n \$_tt_error = '';\n}";
|
|
}
|
|
else {
|
|
$default = '# NO DEFAULT';
|
|
}
|
|
$default = pad($default, 2) if $PRETTY;
|
|
|
|
$handlers = join(', ', @$handlers);
|
|
return <<EOF;
|
|
|
|
# TRY
|
|
$OUTPUT do {
|
|
my \$output = '';
|
|
my (\$_tt_error, \$_tt_handler);
|
|
eval {
|
|
$block
|
|
};
|
|
if (\$@) {
|
|
\$_tt_error = \$context->catch(\$@, \\\$output);
|
|
die \$_tt_error if \$_tt_error->type =~ /^(return|stop)\$/;
|
|
\$stash->set('error', \$_tt_error);
|
|
\$stash->set('e', \$_tt_error);
|
|
if (defined (\$_tt_handler = \$_tt_error->select_handler($handlers))) {
|
|
$catchblock
|
|
}
|
|
$default
|
|
}
|
|
$final
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# throw(\@nameargs) [% THROW foo "bar error" %]
|
|
# # => [ [$type], \@args ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub throw {
|
|
my ($self, $nameargs) = @_;
|
|
my ($type, $args) = @$nameargs;
|
|
my $hash = shift(@$args);
|
|
my $info = shift(@$args);
|
|
$type = shift @$type; # uses same parser production as INCLUDE
|
|
# etc., which allow multiple names
|
|
# e.g. INCLUDE foo+bar+baz
|
|
|
|
if (! $info) {
|
|
$args = "$type, undef";
|
|
}
|
|
elsif (@$hash || @$args) {
|
|
local $" = ', ';
|
|
my $i = 0;
|
|
$args = "$type, { args => [ "
|
|
. join(', ', $info, @$args)
|
|
. ' ], '
|
|
. join(', ',
|
|
(map { "'" . $i++ . "' => $_" } ($info, @$args)),
|
|
@$hash)
|
|
. ' }';
|
|
}
|
|
else {
|
|
$args = "$type, $info";
|
|
}
|
|
|
|
return "\$context->throw($args, \\\$output);";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# clear() [% CLEAR %]
|
|
#
|
|
# NOTE: this is redundant, being hard-coded (for now) into Parser.yp
|
|
#------------------------------------------------------------------------
|
|
|
|
sub clear {
|
|
return "\$output = '';";
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# break() [% BREAK %]
|
|
#
|
|
# NOTE: this is redundant, being hard-coded (for now) into Parser.yp
|
|
#------------------------------------------------------------------------
|
|
|
|
sub OLD_break {
|
|
return 'last LOOP;';
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# return() [% RETURN %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub return {
|
|
return "\$context->throw('return', '', \\\$output);";
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# stop() [% STOP %]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub stop {
|
|
return "\$context->throw('stop', '', \\\$output);";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# use(\@lnameargs) [% USE alias = plugin(args) %]
|
|
# # => [ [$file, ...], \@args, $alias ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub use {
|
|
my ($self, $lnameargs) = @_;
|
|
my ($file, $args, $alias) = @$lnameargs;
|
|
$file = shift @$file; # same production rule as INCLUDE
|
|
$alias ||= $file;
|
|
$args = &args($self, $args);
|
|
$file .= ", $args" if $args;
|
|
# my $set = &assign($self, $alias, '$plugin');
|
|
return "# USE\n"
|
|
. "\$stash->set($alias,\n"
|
|
. " \$context->plugin($file));";
|
|
}
|
|
|
|
#------------------------------------------------------------------------
|
|
# view(\@nameargs, $block) [% VIEW name args %]
|
|
# # => [ [$file, ... ], \@args ]
|
|
#------------------------------------------------------------------------
|
|
|
|
sub view {
|
|
my ($self, $nameargs, $block, $defblocks) = @_;
|
|
my ($name, $args) = @$nameargs;
|
|
my $hash = shift @$args;
|
|
$name = shift @$name; # same production rule as INCLUDE
|
|
$block = pad($block, 1) if $PRETTY;
|
|
|
|
if (%$defblocks) {
|
|
$defblocks = join(",\n", map { "'$_' => $defblocks->{ $_ }" }
|
|
keys %$defblocks);
|
|
$defblocks = pad($defblocks, 1) if $PRETTY;
|
|
$defblocks = "{\n$defblocks\n}";
|
|
push(@$hash, "'blocks'", $defblocks);
|
|
}
|
|
$hash = @$hash ? '{ ' . join(', ', @$hash) . ' }' : '';
|
|
|
|
return <<EOF;
|
|
# VIEW
|
|
do {
|
|
my \$output = '';
|
|
my \$_tt_oldv = \$stash->get('view');
|
|
my \$_tt_view = \$context->view($hash);
|
|
\$stash->set($name, \$_tt_view);
|
|
\$stash->set('view', \$_tt_view);
|
|
|
|
$block
|
|
|
|
\$stash->set('view', \$_tt_oldv);
|
|
\$_tt_view->seal();
|
|
# \$output; # not used - commented out to avoid warning
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# perl($block)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub perl {
|
|
my ($self, $block) = @_;
|
|
$block = pad($block, 1) if $PRETTY;
|
|
|
|
return <<EOF;
|
|
|
|
# PERL
|
|
\$context->throw('perl', 'EVAL_PERL not set')
|
|
unless \$context->eval_perl();
|
|
|
|
$OUTPUT do {
|
|
my \$output = "package Template::Perl;\\n";
|
|
|
|
$block
|
|
|
|
local(\$Template::Perl::context) = \$context;
|
|
local(\$Template::Perl::stash) = \$stash;
|
|
|
|
my \$_tt_result = '';
|
|
tie *Template::Perl::PERLOUT, 'Template::TieString', \\\$_tt_result;
|
|
my \$_tt_save_stdout = select *Template::Perl::PERLOUT;
|
|
|
|
eval \$output;
|
|
select \$_tt_save_stdout;
|
|
\$context->throw(\$@) if \$@;
|
|
\$_tt_result;
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# no_perl()
|
|
#------------------------------------------------------------------------
|
|
|
|
sub no_perl {
|
|
my $self = shift;
|
|
return "\$context->throw('perl', 'EVAL_PERL not set');";
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# rawperl($block)
|
|
#
|
|
# NOTE: perhaps test context EVAL_PERL switch at compile time rather than
|
|
# runtime?
|
|
#------------------------------------------------------------------------
|
|
|
|
sub rawperl {
|
|
my ($self, $block, $line) = @_;
|
|
for ($block) {
|
|
s/^\n+//;
|
|
s/\n+$//;
|
|
}
|
|
$block = pad($block, 1) if $PRETTY;
|
|
$line = $line ? " (starting line $line)" : '';
|
|
|
|
return <<EOF;
|
|
# RAWPERL
|
|
#line 1 "RAWPERL block$line"
|
|
$block
|
|
EOF
|
|
}
|
|
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# filter()
|
|
#------------------------------------------------------------------------
|
|
|
|
sub filter {
|
|
my ($self, $lnameargs, $block) = @_;
|
|
my ($name, $args, $alias) = @$lnameargs;
|
|
$name = shift @$name;
|
|
$args = &args($self, $args);
|
|
$args = $args ? "$args, $alias" : ", undef, $alias"
|
|
if $alias;
|
|
$name .= ", $args" if $args;
|
|
$block = pad($block, 1) if $PRETTY;
|
|
|
|
return <<EOF;
|
|
|
|
# FILTER
|
|
$OUTPUT do {
|
|
my \$output = '';
|
|
my \$_tt_filter = \$context->filter($name)
|
|
|| \$context->throw(\$context->error);
|
|
|
|
$block
|
|
|
|
&\$_tt_filter(\$output);
|
|
};
|
|
EOF
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# capture($name, $block)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub capture {
|
|
my ($self, $name, $block) = @_;
|
|
|
|
if (ref $name) {
|
|
if (scalar @$name == 2 && ! $name->[1]) {
|
|
$name = $name->[0];
|
|
}
|
|
else {
|
|
$name = '[' . join(', ', @$name) . ']';
|
|
}
|
|
}
|
|
$block = pad($block, 1) if $PRETTY;
|
|
|
|
return <<EOF;
|
|
|
|
# CAPTURE
|
|
\$stash->set($name, do {
|
|
my \$output = '';
|
|
$block
|
|
\$output;
|
|
});
|
|
EOF
|
|
|
|
}
|
|
|
|
|
|
#------------------------------------------------------------------------
|
|
# macro($name, $block, \@args)
|
|
#------------------------------------------------------------------------
|
|
|
|
sub macro {
|
|
my ($self, $ident, $block, $args) = @_;
|
|
$block = pad($block, 2) if $PRETTY;
|
|
|
|
if ($args) {
|
|
my $nargs = scalar @$args;
|
|
$args = join(', ', map { "'$_'" } @$args);
|
|
$args = $nargs > 1
|
|
? "\@_tt_args{ $args } = splice(\@_, 0, $nargs)"
|
|
: "\$_tt_args{ $args } = shift";
|
|
|
|
return <<EOF;
|
|
|
|
# MACRO
|
|
\$stash->set('$ident', sub {
|
|
my \$output = '';
|
|
my (%_tt_args, \$_tt_params);
|
|
$args;
|
|
\$_tt_params = shift;
|
|
\$_tt_params = { } unless ref(\$_tt_params) eq 'HASH';
|
|
\$_tt_params = { \%_tt_args, %\$_tt_params };
|
|
|
|
my \$stash = \$context->localise(\$_tt_params);
|
|
eval {
|
|
$block
|
|
};
|
|
\$stash = \$context->delocalise();
|
|
die \$@ if \$@;
|
|
return \$output;
|
|
});
|
|
EOF
|
|
|
|
}
|
|
else {
|
|
return <<EOF;
|
|
|
|
# MACRO
|
|
\$stash->set('$ident', sub {
|
|
my \$_tt_params = \$_[0] if ref(\$_[0]) eq 'HASH';
|
|
my \$output = '';
|
|
|
|
my \$stash = \$context->localise(\$_tt_params);
|
|
eval {
|
|
$block
|
|
};
|
|
\$stash = \$context->delocalise();
|
|
die \$@ if \$@;
|
|
return \$output;
|
|
});
|
|
EOF
|
|
}
|
|
}
|
|
|
|
|
|
sub debug {
|
|
my ($self, $nameargs) = @_;
|
|
my ($file, $args) = @$nameargs;
|
|
my $hash = shift @$args;
|
|
$args = join(', ', @$file, @$args);
|
|
$args .= @$hash ? ', { ' . join(', ', @$hash) . ' }' : '';
|
|
return "$OUTPUT \$context->debugging($args); ## DEBUG ##";
|
|
}
|
|
|
|
|
|
1;
|
|
|
|
__END__
|
|
|
|
=head1 NAME
|
|
|
|
Template::Directive - Perl code generator for template directives
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
# no user serviceable parts inside
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
The C<Template::Directive> module defines a number of methods that
|
|
generate Perl code for the runtime representation of the various
|
|
Template Toolkit directives.
|
|
|
|
It is used internally by the L<Template::Parser> module.
|
|
|
|
=head1 AUTHOR
|
|
|
|
Andy Wardley E<lt>abw@wardley.orgE<gt> L<http://wardley.org/>
|
|
|
|
=head1 COPYRIGHT
|
|
|
|
Copyright (C) 1996-2007 Andy Wardley. All Rights Reserved.
|
|
|
|
This module is free software; you can redistribute it and/or
|
|
modify it under the same terms as Perl itself.
|
|
|
|
=head1 SEE ALSO
|
|
|
|
L<Template::Parser>
|
|
|
|
=cut
|
|
|
|
# Local Variables:
|
|
# mode: perl
|
|
# perl-indent-level: 4
|
|
# indent-tabs-mode: nil
|
|
# End:
|
|
#
|
|
# vim: expandtab shiftwidth=4:
|
|
|