X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FReaction%2FUI%2FLayoutSet.pm;h=792bb701036de91e695d631a1c1b1d1f248177c8;hb=cb92a3a388a813d1309757155a4b7750eb9f5504;hp=793568fd37127e599f5b9fec1f4c41ccb6d64821;hpb=7adfd53f17f66ffe93763e944ed1d3fc52a369dc;p=catagits%2FReaction.git diff --git a/lib/Reaction/UI/LayoutSet.pm b/lib/Reaction/UI/LayoutSet.pm index 793568f..792bb70 100644 --- a/lib/Reaction/UI/LayoutSet.pm +++ b/lib/Reaction/UI/LayoutSet.pm @@ -5,48 +5,107 @@ use File::Spec; class LayoutSet which { - has 'fragments' => (is => 'ro', default => sub { {} }); + has 'layouts' => (is => 'ro', default => sub { {} }); has 'name' => (is => 'ro', required => 1); has 'source_file' => (is => 'rw', lazy_fail => 1); + has 'file_extension'=> (isa => 'Str', is => 'rw', lazy_build => 1); + + has 'widget_class' => ( + is => 'rw', lazy_fail => 1, predicate => 'has_widget_class' + ); + + has 'super' => (is => 'rw', predicate => 'has_super'); + + implements _build_file_extension => as { 'html' }; implements 'BUILD' => as { my ($self, $args) = @_; my @path = @{$args->{search_path}||[]}; confess "No search_path provided" unless @path; + confess "No view object provided" unless $args->{view}; my $found; + my $ext = $self->file_extension; SEARCH: foreach my $path (@path) { - my $cand = $path->file($self->name); + my $cand = $path->file($self->name . ".${ext}"); + #print STDERR $cand,"\n"; if ($cand->stat) { - $self->_load_file($cand); + $self->_load_file($cand, $args); $found = 1; last SEARCH; } } confess "Unable to load file for LayoutSet ".$self->name unless $found; + unless ($self->has_widget_class) { + $self->widget_class($args->{view}->widget_class_for($self)); + } + }; + + implements 'widget_order_for' => as { + my ($self, $name) = @_; + return ( + ($self->has_layout($name) + ? ([ $self->widget_class, $self ]) #; + : ()), + ($self->has_super + ? ($self->super->widget_order_for($name)) + : ()), + ); + }; + + implements 'layout_names' => as { + my ($self) = @_; + my %seen; + return [ + grep { !$seen{$_}++ } + keys %{shift->layouts}, + ($self->has_super + ? (@{$self->super->layout_names}) + : ()) + ]; }; + implements 'has_layout' => as { exists $_[0]->layouts->{$_[1]} }; + implements '_load_file' => as { - my ($self, $file) = @_; + my ($self, $file, $build_args) = @_; my $data = $file->slurp; - my $fragments = $self->fragments; - # cheesy match for "=for layout fragmentname ... =something" + my $layouts = $self->layouts; + # cheesy match for "=for layout name ... =something" # final split group also handles last in file, (?==) is lookahead - # assertion for '=' so "=for layout fragment1 ... =for layout fragment2" - # doesn't have the match pos go past the latter = and lose fragment2 - while ($data =~ m/=for layout (.*?)\n(.+?)(?:\n(?==)|$)/sg) { - my ($fname, $text) = ($1, $2); - $fragments->{$fname} = $text; + # assertion for '=' so "=for layout name1 ... =for layout name2" + # doesn't have the match pos go past the latter = and lose name2 + while ($data =~ m/=(.*?)\n(.*?)(?:\n(?==)|$)/sg) { + my ($data, $text) = ($1, $2); + if ($data =~ /^for layout (\S+)/) { + my $fname = $1; + #remove extra whitespace without killing indentation + #remove all empty leading lines. and trailing whitespace + ($layouts->{$fname}) = + ($text =~ /^(?:\s*\n)*((?:.*?\n)*(?:.*?\S+.*?\n))\s*$/m); + } elsif ($data =~ /^extends (\S+)/) { + my $super_name = $1; + $self->super($build_args->{view}->create_layout_set($super_name)) + } elsif ($data =~ /^cut/) { + # no-op + } else { + confess "Unparseable directive ${data}"; + } } $self->source_file($file); }; implements 'widget_type' => as { my ($self) = @_; - return join('', map { ucfirst($_) } split('_', $self->name)); + my $widget = join('', map { ucfirst($_) } split('_', $self->name)); + $widget = join('::', map { ucfirst($_) } split('/', $widget)); + + #print STDERR "--- ", $self->name, " maps to widget $widget \n"; + + return $widget; }; - + }; 1;