package Reaction::UI::Controller::Collection; use strict; use warnings; use base 'Reaction::UI::Controller'; use Reaction::Class; use aliased 'Reaction::UI::ViewPort::Collection::Grid'; use aliased 'Reaction::UI::ViewPort::Object'; has model_name => (isa => 'Str', is => 'rw', required => 1); has collection_name => (isa => 'Str', is => 'rw', required => 1); has action_viewport_map => (isa => 'HashRef', is => 'rw', lazy_build => 1); has action_viewport_args => (isa => 'HashRef', is => 'rw', lazy_build => 1); has default_member_actions => ( isa => 'ArrayRef', is => 'rw', lazy_build => 1 ); has default_collection_actions => ( isa => 'ArrayRef', is => 'rw', lazy_build => 1 ); sub _build_default_member_actions { ['view'] } sub _build_default_collection_actions { [] } sub _build_action_viewport_map { my $self = shift; my %map; $map{list} = Grid; $map{view} = Object; #if grep {$_ eq 'view'} @{$self->default_member_actions}; return \%map; } sub _build_action_viewport_args { my $self = shift; my $args = { list => { Member => {} } }; my $m_protos = $args->{list}{Member}{action_prototypes} = {}; for my $action_name( @{ $self->default_member_actions }){ my $label = join(' ', map { ucfirst } split(/_/, $action_name)); my $proto = $self->_build_member_action_prototype($label, $action_name); $m_protos->{$action_name} = $proto; } my $c_protos = $args->{list}{action_prototypes} = {}; for my $action_name( @{ $self->default_collection_actions }){ my $label = join(' ', map { ucfirst } split(/_/, $action_name)); my $proto = $self->_build_collection_action_prototype($label, $action_name); $c_protos->{$action_name} = $proto; } return $args; } sub _build_member_action_prototype { my ($self, $label, $action_name) = @_; return { label => $label, uri => sub { my $action = $self->action_for($action_name); $_[1]->uri_for($action, [ @{$_[1]->req->captures}, $_[0]->__id ]); }, }; } sub _build_collection_action_prototype { my ($self, $label, $action_name) = @_; return { label => $label, uri => sub { my $action = $self->action_for($action_name); $_[1]->uri_for($action, $_[1]->req->captures); }, }; } #XXX candidate for futre optimization, should cache reader? sub get_collection { my ($self, $c) = @_; my $model = $c->model( $self->model_name ); confess "Failed to find Catalyst model named: " . $self->model_name unless $model; my $collection = $self->collection_name; if( my $meth = $model->can( $collection ) ){ return $model->$meth; } elsif ( my $meta = $model->can('meta') ){ if ( my $attr = $model->$meta->find_attribute_by_name($collection) ) { my $reader = $attr->get_read_method; return $model->$reader; } } confess "Failed to find collection $collection"; } sub base :Action :CaptureArgs(0) { my ($self, $c) = @_; } sub object :Chained('base') :PathPart('id') :CaptureArgs(1) { my ($self, $c, $key) = @_; my $object = $self->get_collection($c)->find($key); $c->detach("/error_404") unless $object; $c->stash(object => $object); } sub list :Chained('base') :PathPart('') :Args(0) { my ($self, $c) = @_; my $collection = $c->stash->{collection} || $self->get_collection($c); $self->basic_page($c, { collection => $collection }); } sub view :Chained('object') :Args(0) { my ($self, $c) = @_; $self->basic_page($c, { model => $c->stash->{object} }); } sub basic_page { my ($self, $c, $vp_args) = @_; my $action_name = $c->stack->[-1]->name; my $vp = $self->action_viewport_map->{$action_name}, my $args = $self->merge_config_hashes ( $vp_args || {}, $self->action_viewport_args->{$action_name} || {} , ); return $self->push_viewport($vp, %$args); } 1; __END__; =head1 NAME Reaction::UI::Controller =head1 DESCRIPTION Controller class used to make displaying collections easier. Inherits from L. =head1 ATTRIBUTES =head2 model_name The name of the model this controller will use as it's data source. Should be a name that can be passed to C<$C-Emodel> =head2 collection_name The name of the collection whithin the model that this Controller will be utilizing. =head2 action_viewport_map =over 4 =item B<_build_action_viewport_map> - Provided builder method, see METHODS =item B - Auto generated predicate =item B- Auto generated clearer method =back Read-write lazy building hashref. The keys should match action names in the Controller and the value should be the ViewPort class that this action should use. See method C for more info. =head2 action_viewport_args Read-write lazy building hashref. Additional ViewPort arguments for the action named as the key in the controller. See method C for more info. =over 4 =item B<_build_action_viewport_args> - Provided builder method, see METHODS =item B - Auto generated predicate =item B- Auto generated clearer method =back =head2 default_member_actions Read-write lazy building arrayref. The names of the member actions (the actions that apply to each member of the collection and typically have an object as a target e.g. update,delete) to be enabled by default. By default, this is only 'view' =over 4 =item B<_build_default_member_actions> - Provided builder method, see METHODS =item B - Auto generated predicate =item B- Auto generated clearer method =back =head2 default_collection_actions Read-write lazy building arrayref. The names of the collection actions (the actions that apply to the entire collection and typically have a collection as a target e.g. create, delete_all) to be enabled by default. By default, this is only empty. =over 4 =item B<_build_default_member_actions> - Provided builder method, see METHODS =item B - Auto generated predicate =item B- Auto generated clearer method =back =head1 METHODS =head2 get_collection $c Returns an instance of the collection this controller uses. =head2 _build_action_viewport_map Provided builder for C. Returns a hash containing: list => 'Reaction::UI::ViewPort::Collection::Grid', view => 'Reaction::UI::ViewPort::Object', =head2 _build_action_viewport_args By default will reurn a hashref containing action prototypes for all default member and collection actions. The prototype URI generators are generated by C<_build_member_action_prototype> and C<_build_collection_action_prototype> respectively and labels are the result of replacing underscores in the name with spaces and capitalizing the first letter. If you plan to use custom actions that are not supported by this scheme or you would like to customize the values it is suggested you wrap / override this method. Default output for a controller having only 'view' enabled: { list => { action_prototypes => {}, Member => { action_prototypes => { view => {label => 'View', uri => sub{...} }, }, }, }, } =head2 _build_member_action_prototype $label, $action_name Creates an action prototype suitable for creating action links in L. C<$action_name> should be the name of a Catalyst action in this controller.The prototype will generate a URI based on the action, current captures. =head2 _build_collection_action_prototype $label, $action_name =head2 basic_page $c, \%vp_args Accepts two arguments, context, and a hashref of viewport arguments. It will automatically determine the action name using the catalyst stack and call C with the ViewPort class name contained in the C with a set of options determined by merging C<$vp_args> and the arguments contained in C, if any. =head1 ACTIONS =head2 base Chain link, no-op. =head2 list Chain link, chained to C. C fetches the collection for the model and calls C with a single argument, C. The default ViewPort for this action is C and can be changed by altering the C attribute hash. =head2 object Chain link, chained to C, captures one argument, 'id'. Attempts to find a single object by searching for a member of the current collection which has a Primary Key or Unique constraint matching that argument. If the object is found it is stored in the stash under the C key. =head2 view Chain link, chained to C. Calls C with one argument, C, which contains an instance of the object fetched by the C action link. The default ViewPort for this action is C and can be changed by altering the C attribute hash. =head1 SEE ALSO L =head1 AUTHORS See L for authors. =head1 LICENSE See L for the license. =cut