1 package Reaction::UI::ViewPort::Role::Actions;
4 use Reaction::UI::ViewPort::URI;
6 use namespace::clean -except => [ qw(meta) ];
19 has action_filter => (
20 isa => 'CodeRef', is => 'ro',
23 has action_prototypes => (
30 has computed_action_order => (
36 sub _filter_action_list {
38 my $actions = [keys %{$self->action_prototypes}];
39 return $self->has_action_filter ?
40 $self->action_filter->($actions, $self->model)
44 sub _build_computed_action_order {
46 my $ordered = $self->sort_by_spec(
47 ($self->has_action_order ? $self->action_order : []),
48 $self->_filter_action_list
57 my $loc = $self->location;
58 my $target = $self->model;
60 foreach my $proto_name ( @{ $self->computed_action_order } ) {
61 my $proto = $self->action_prototypes->{$proto_name};
62 my $uri = $proto->{uri} or confess('uri is required in prototype action');
63 my $label = exists $proto->{label} ? $proto->{label} : $proto_name;
64 my $layout = exists $proto->{layout} ? $proto->{layout} : 'uri';
65 my $layout_args = exists $proto->{layout_args} ? $proto->{layout_args} : {};
67 my $action = Reaction::UI::ViewPort::URI->new(
68 location => join ('-', $loc, 'action', $i++),
69 uri => ( ref($uri) eq 'CODE' ? $uri->($target, $ctx) : $uri ),
70 display => ( ref($label) eq 'CODE' ? $label->($target, $ctx) : $label ),
71 layout => ( ref($layout) eq 'CODE' ? $layout->($target, $ctx) : $layout ),
72 layout_args => ( ref($layout_args) eq 'CODE' ? $layout_args->($target, $ctx) : $layout_args ),
85 Reaction::UI::ViewPort::Role::Actions
89 A role to ease attaching actions to L<Reaction::InterfaceModel::Object>s
95 Read-only, lazy-building ArrayRef of URI objects pointing to actions.
97 =head2 action_prototypes
99 A HashRef of prototypes for building the Action links. The prototypes should be
102 my %action_prototypes = (
103 example_action => { label => 'Example Action', uri => $uri_obj },
106 #or you can get fancy and do something like what is below:
108 my($im, $ctx) = @_; #InterfaceModel::Object/Collection, Catalyst Context
112 my($im, $ctx) = @_; #InterfaceModel::Object/Collection, Catalyst Context
113 return return $ctx->uri_for('some_action');
115 my %action_prototypes = (
116 example_action => { label => \&make_label, uri => \&make_uri },
121 User-provided ArrayRef with how the actions should be ordered eg
123 action_order => [qw/view edit delete/]
125 =head2 computed_action_order
127 Read-only lazy-building ARRAY ref. The final computed action order. This may
128 differ from the C<action_order> provided if you any actions were not included
133 =head2 _build_actions
135 Cycle through the C<computed_action_order> and create a new
136 L<ViewPort::URI|Reaction::UI::ViewPort::URI> object for each action using the
139 =head2 _build_computed_action_order
141 Compute the final action ordering by using the provided C<action_order> as a
142 spec to order all the present actions (the keys of C<action_prototypes>)
144 =head1 ACTION PROTOTYPES
146 Action prototypes are simply hashrefs that must contain a C<uri> key and may
147 contain a C<label> key. The label can be anything that the display attribute of
148 L<ViewPort::URI|Reaction::UI::ViewPort::URI> will accept, usually a scalar or a
149 ViewPort. The value for C<uri> may be either a scalar, a L<URI> object (or
150 anything that C<ISA URI>).
152 Additionally, both C<label> and C<uri> can be CODE refs. In this case, the code
153 will be executed at C<_build_actions> time and will recieve two arguments, the
154 value returned by C<model> and the value returned by C<ctx> in that order. Both
155 of these methods should be implemented in the consuming class. By convention,
156 model refers to the target of the action, an C<InterfaceModel::Object> in the
157 case of a member action and an C<InterfaceModel::Collection> in the case of a
158 Collection action. C<ctx> should be the current Catalyst context.
162 See L<Reaction::Class> for authors.
166 See L<Reaction::Class> for the license.