1 package Reaction::UI::Controller;
3 use base qw(Catalyst::Controller); # Reaction::Object);
6 use Scalar::Util 'weaken';
7 use namespace::clean -except => [ qw(meta) ];
9 has context => (is => 'ro', isa => 'Object', weak_ref => 1);
10 with 'Catalyst::Component::InstancePerContext';
12 sub build_per_context_instance {
13 my ($self, $c, @args) = @_;
14 my $class = ref($self) || $self;
15 my $newself = $class->new($self->_application, {%{$self || {}}, context => $c, @args});
21 my $c = $self->context;
22 my $focus_stack = $c->stash->{focus_stack};
23 my ($class, @proto_args) = @_;
25 if (my $vp_attr = $c->stack->[-1]->attributes->{ViewPort}) {
26 if (ref($vp_attr) eq 'ARRAY') {
27 $vp_attr = $vp_attr->[0];
29 if (ref($vp_attr) eq 'HASH') {
30 $class = $vp_attr->{class} if defined $vp_attr->{class};
31 %args = %{ $self->merge_config_hashes($vp_attr, {@proto_args}) };
42 if (exists $args{next_action} && !ref($args{next_action})) {
43 $args{next_action} = [ $self, 'redirect_to', $args{next_action} ];
45 $focus_stack->push_viewport($class, %args);
49 return shift->context->stash->{focus_stack}->pop_viewport;
52 sub pop_viewports_to {
54 return $self->context->stash->{focus_stack}->pop_viewports_to($vp);
58 my ($self, $c, $to, $cap, $args, $attrs) = @_;
60 #the confess calls could be changed later to $c->log ?
62 my $reftype = ref($to);
64 $action = $self->action_for($to);
65 confess("Failed to locate action ${to} in " . blessed($self)) unless $action;
66 } elsif($reftype eq 'ARRAY' && @$to == 2){ #is that overkill / too strict?
67 $action = $c->controller($to->[0])->action_for($to->[1]);
68 confess("Failed to locate action $to->[1] in $to->[0]" ) unless $action;
69 } elsif( blessed $to && $to->isa('Catalyst::Action') ){
72 confess("Failed to locate action from ${to}");
75 $cap ||= $c->req->captures;
76 $args ||= $c->req->args;
78 my $uri = $c->uri_for($action, $cap, @$args, $attrs);
79 $c->res->redirect($uri);
82 sub make_context_closure {
83 my($self, $closure) = @_;
84 my $ctx = $self->context;
86 return sub { $closure->($ctx, @_) };
95 Reaction::UI::Controller - Reaction Base Controller Class
99 package MyApp::Controller::Foo;
102 use parent 'Reaction::UI::Controller';
104 use aliased 'Reaction::UI::ViewPort';
106 sub foo: Chained('/base') Args(0) {
107 my ($self, $ctx) = @_;
109 $ctx->push_viewport(ViewPort,
118 Base Reaction Controller class. Inherits from:
122 =item L<Catalyst::Controller>
123 =item L<Catalyst::Component::ACCEPT_CONTEXT>
124 =item L<Reaction::Object>
130 =head2 push_viewport $vp_class, %args
132 Creates a new instance of the L<Reaction::UI::ViewPort> class
133 ($vp_class) using the rest of the arguments given (%args). Defaults of
134 the action can be overridden by using the C<ViewPort> key in the
135 controller configuration. For example to override the default number
136 of items in a CRUD list action:
140 list => { ViewPort => { per_page => 50 } },
144 The ViewPort is added to the L<Reaction::UI::Window>'s FocusStack in
145 the stash, and also returned to the calling code.
151 =item L<Reaction::UI::Controller::Root>
152 =item L<Reaction::UI::Window>
156 TODO: explain how next_action as a scalar gets converted to the redirect arrayref thing
160 =head2 pop_viewport_to $vp
162 Call L<Reaction::UI::FocusStack/pop_viewport> or
163 L<Reaction::UI::FocusStack/pop_viewport_to> on
164 the C<< $c->stash->{focus_stack} >>.
166 =head2 redirect_to $c, $to, $captures, $args, $attrs
168 Construct a URI and redirect to it.
174 =item The name of an action in the current controller.
176 =item A L<Catalyst::Action> instance.
178 =item An arrayref of controller name and the name of an action in that
183 $captures and $args default to the current requests $captures and
184 $args if not supplied.
186 =head2 make_context_closure
188 The purpose of this method is to prevent memory leaks.
189 It weakens the context object, often denoted $c, and passes it as the
190 first argument to the sub{} that is passed to the make_context_closure method.
195 make_context_closure returns sub { $sub_you_gave_it->($weak_c, @_)
199 To further expound up this useful construct consider code written before
200 make_context_closure was created:
204 $self->after_search( $c, @_ );
208 This could be rewritten as:
210 on_apply_callback => $self->make_context_closure(
213 $self->after_search( $weak_c, @_ );
217 Or even more succintly:
219 on_apply_callback => $self->make_context_closure(
221 $self->after_search( @_ );
227 See L<Reaction::Class> for authors.
231 See L<Reaction::Class> for the license.