| Jifty documentation | Contained in the Jifty distribution. |
Jifty::Web - Web framework for a Jifty application
Creates a new Jifty::Web object
Returns a HTML::Mason::Request object
Send a string to the browser.
Returns the root url of this Jifty application. This is pulled from the configuration file. Takes an optional named path which will form the path part of the resulting URL.
Returns a unique identifier, guaranteed to be unique within the runtime of a particular process (i.e., within the lifetime of Jifty.pm). There's no sort of global uniqueness guarantee, but it should be good enough for generating things like moniker names.
Sets up the current session object (a Jifty::Web::Session tied
hash). Aborts if the session is already loaded.
Returns the current session's hash. In a regular user environment, it persists, but a request can stop that by handing it a regular hash to use.
Getter/setter for the current user; this gets or sets the 'user' key in the session. These are Jifty::Record objects.
If a temporary_current_user has been set, will return that instead.
If the current application has no loaded current user, we get an empty
app-specific CurrentUser object. (This
ApplicationClass::CurrentUser class, a subclass of
Jifty::CurrentUser, is autogenerated if it doesn't exist).
Sets the current request's current_user to USER if set.
This value will _not_ be persisted to the session at the end of the request. To restore the original value, set temporary_current_user to undef.
This method sets up a current session, and then processes the given Jifty::Request object. If no request object is given, processes the request object in request.
Each action on the request is vetted in three ways -- first, it must
be marked as active by the Jifty::Request (this is the default).
Second, it must be in the set of allowed classes of actions (see
is_allowed in Jifty::API). Finally, the action must validate. If it
passes all of these criteria, the action is fit to be run.
Before they are run, however, the request has a chance to be interrupted and saved away into a continuation, to be resumed at some later point. This is handled by save_continuation in Jifty::Request.
If the continuation isn't being saved, then handle_request goes on
to run all of the actions. If all of the actions are successful, it
looks to see if the request wished to call any continuations, possibly
jumping back and re-running a request that was interrupted in the
past. This is handled by call_continuation in Jifty::Request.
For more details about continuations, see Jifty::Continuation.
Gets or sets the current Jifty::Request object.
Gets or sets the current Jifty::Response object.
Returns the current Jifty::Web::Form object, creating one if there isn't one already.
Creates a new action (an instance of a subclass of
Jifty::Action). The named arguments passed to this method are
passed on to the new method of the action named in CLASS.
If you pass an odd number of arguments, then the first argument
is interpreted as class.
CLASS is qualified, and an instance of that
class is created, passing the Jifty::Web object, the MONIKER,
and any other arguments that new_action was supplied.
MONIKER is a unique designator of an action on a page. The moniker
is content-free and non-fattening, and may be auto-generated. It is
used to tie together arguments that relate to the same action.
ORDER defines the order in which the action is run, with lower
numerical values running first.
ARGUMENTS are passed to the new in Jifty::Action method. In
addition, if the current request ($self->request) contains an
action with a matching moniker, any arguments that are in that
requested action but not in the PARAMHASH list are set. This
implements "sticky fields".
As a contrast to add_action in Jifty::Web::Form, this does not add the action to the current form -- instead, the first form field to be rendered will automatically register the action in the current form field at that time.
Given a Jifty::Request::Action, creates a new action using new_action.
Returns an array of Jifty::Action objects, one for each Jifty::Request::Action that is marked as failed in the current response.
As failed_actions, but for actions that completed successfully; less often used.
Gets or sets the next page for the framework to show. This is
normally set during the take_action method or a Jifty::Action
Gets or sets whether we should force a redirect to next_page, even
if it's already the current page. You might set this, e.g. to force a
redirect after a POSTed action.
Returns true if we need to redirect, now that we've processed all the
actions. We need a redirect if either next_page is different from
the current page, or force_redirect has been set.
Handle redirection inside webservices call. This is meant to be hooked by plugin that knows what to do with it.
Redirect to the next page. If you pass this method a parameter, it redirects to that URL rather than next_page.
It creates a continuation of where you want to be, and then calls it. If you want to redirect to a page with parameters, pass in a Jifty::Web::Form::Clickable object.
Returns the Jifty::Request of our enclosing continuation, or an empty Jifty::Request if we are not in a continuation.
If called in non-void context, creates and renders a Jifty::Web::Form::Clickable with the given PARAMHASH, forcing a continuation save.
In void context, does a redirect to the URL that the Jifty::Web::Form::Clickable object generates.
Both of these versions preserve all state variables by default.
Does an instant redirect to the url generated by the Jifty::Web::Form::Clickable object generated by the PARAMHASH.
Generates and renders a Jifty::Web::Form::Clickable using the given PARAMHASH.
If called in non-void context, creates and renders a Jifty::Web::Form::Clickable using the given PARAMHASH, additionally defaults to calling the current continuation.
Takes an additional argument, to, which can specify a default path
to return to if there is no current continuation.
In void context, does a redirect to the URL that the Jifty::Web::Form::Clickable object generates.
Returns true if PATH is a valid template inside your template root. This checks for both Template::Declare and HTML::Mason Templates. Specifically, returns a reference to the handler which can process the template.
If PATH is a reference to the path, it will update the path to
append /index.html if the path in question doesn't exist, but the
index does.
Use our templating system to render a template. Searches through view_handlers in Jifty::Handler to find the first handler which provides the given template, and caches the handler for future requests.
Catches errors, and redirects to /errors/500; also shows
/errors/404 if the template cannot be found.
Outputs any messages that have been added, in <div id="messages"> and <div id="errors"> tags. Messages are added by calling message in Jifty::Result.
If a moniker is specified, only messages for that moniker are rendered.
Render success messages for the given moniker, or all of them if no moniker is given.
Render error messages for the given moniker, or all of them if no moniker is given.
Output any messages of the given TYPE (either 'error' or 'message') in a <div id="TYPEs"> tag. If a moniker is given, only renders messages for that action. Internal helper for render_success_messages and render_errors.
Returns an URL-encoded query string piece representing the arguments passed to it.
HTML-escapes the given string and returns it
URI-escapes the given string and returns it
Returns a <link> tag for the compressed CSS
Pushes files onto Jifty->web->css_files
Returns a <script> tag for the compressed Javascript.
Your application specific javascript goes in share/web/static/js/app.js. This will be automagically included if it exists.
If you want to add javascript behaviour to your page using CSS
selectors then put your behaviour rules in
share/web/static/js/app_behaviour.js which will also be
automagically included if it exists. The behaviour.js library is
included by Jifty. For more information on behaviour.js see
http://bennolan.com/behaviour/.
However if you want to include other javascript libraries you need to
add them to the javascript_libs array of your application. Do this in
the start sub of your main application class. For example if your
application is Foo then in Foo.pm in lib
sub start {
Jifty->web->add_javascript(qw( jslib1.js jslib2.js ) );
}
The add_javascript method will append the files to javascript_libs. If you need a different order, you'll have to massage javascript_libs directly.
Jifty will look for javascript libraries under share/web/static/js/ by default as well as any plugin static roots.
Pushes files onto Jifty->web->javascript_libs
Removes the given files from Jifty->web->javascript_libs.
This is intended for plugins or applications that provide another version of the functionality given in our default JS libraries. For example, the CSSQuery plugin will get rid of the cssQuery-jQuery.js back-compat script.
Pushes URLs onto Jifty->web->external_javascript_libs
Gets a page specific variable from the request object.
Takes a key-value pair for variables to serialize and hand off to the next page.
Behind the scenes, these variables get serialized into every link or form that is marked as 'state preserving'. See Jifty::Web::Form::Clickable.
Passing undef as a value will remove the variable
Returns all of the state variables that have been set for the next request, as a hash;
N.B. These are not prefixed with J:V-, as they were in earlier
versions of Jifty
Remove all the state variables to be serialized for the next request.
Given a fully QUALIFIED NAME of a region, returns the
Jifty::Web::PageRegion with that name, or undef if no such region
exists.
The provided PARAMHASH is used to create and render a
Jifty::Web::PageRegion; the PARAMHASH is passed directly to its
new in Jifty::Web::PageRegion method, and then
render in Jifty::Web::PageRegion is called.
Replaces the current region with a new region and renders it Returns undef if there's no current region
Returns the current Jifty::Web::PageRegion, or undef if there is none.
Returns the fully qualified name of the current
Jifty::Web::PageRegion, or the empty string if there is none. If
REGION is supplied, gives the qualified name of REGION were it
placed in the current region. You may also use a literal region name.
Returns the fully qualified name of the current
Jifty::Web::PageRegion's parent, or the empty string if there is none. If
REGION is supplied, gives the qualified name of REGION were it
placed in the parent region. You may also use a literal region name.
Indicates whether the current request was made using SSL.
| Jifty documentation | Contained in the Jifty distribution. |
use warnings; use strict; package Jifty::Web;
use XML::Writer; use CSS::Squish; use Digest::MD5 qw(md5_hex); use Scalar::Util qw(blessed); use Carp qw(carp); use base qw/Class::Accessor::Fast Class::Data::Inheritable Jifty::Object/; use vars qw/$SERIAL @JS_INCLUDES/; __PACKAGE__->mk_accessors( qw(next_page force_redirect request response session temporary_current_user) ); __PACKAGE__->mk_classdata($_) for qw(cached_css cached_css_digest cached_css_time css_files javascript_libs external_javascript_libs); __PACKAGE__->css_files([qw( main.css )]); __PACKAGE__->external_javascript_libs([]); __PACKAGE__->javascript_libs([qw( json.js jquery-1.4.1.js iutil.js jifty_interface.js jquery_noconflict.js jquery.jgrowl.js facebox.js behaviour.js jifty.js jifty_utils.js jifty_subs.js jifty_smoothscroll.js calendar.js datetime.js dom-drag.js halo.js combobox.js key_bindings.js context_menu.js yui/yahoo.js yui/dom.js yui/event.js yui/calendar.js yui/element-beta.js yui/tabview.js yui/container.js yui/menu.js app.js app_behaviour.js css_browser_selector.js cssQuery-jquery.js jquery.timepickr.js jquery.ajaxQueue.js jquery.bgiframe.min.js jquery.autocomplete.js ui.core.js ui.sortable.js ordered-list.js uploads.js )]); use Class::Trigger;
sub new { my $class = shift; my $self = bless {region_stack => []}, $class; $self->session(Jifty::Web::Session->new()); $self->clear_state_variables; return ($self); }
sub mason { use HTML::Mason::Request; return HTML::Mason::Request->instance; }
sub out { shift; Jifty->handler->buffer->append(@_); }
sub url { my $self = shift; my %args = (scheme => undef, path => undef, @_); my $uri; my $req = Jifty->web->request; if ($req && $req->uri->host) { $uri = $req->uri->clone; $uri->path_query('/'); } else { $uri = URI->new(Jifty->config->framework("Web")->{BaseURL}); $uri->port(Jifty->config->framework("Web")->{Port}); } if (defined (my $path = $args{path})) { # strip off leading '/' because ->canonical provides one $path =~ s{^/}{}; $uri->path_query($path); } # https is sticky $uri->scheme('https') if $uri->scheme eq 'http' && Jifty->web->is_ssl; # If we're generating a URL from an email (really a Jifty::Notification # subclass), default to http my $level = 0; while ( my $class = caller($level++) ) { if ( $class->isa("Jifty::Notification") ) { $uri->scheme('http'); last; } } $uri->scheme( $args{'scheme'} ) if defined $args{'scheme'}; return $uri->canonical->as_string; }
sub serial { my $class = shift; # We don't use a lexical for the serial number, because then it # would be reset on module refresh $SERIAL ||= 0; return join('', "S", ++$SERIAL, $$ ); # Start at 1. }
# Create the Jifty::Web::Session object sub setup_session { my $self = shift; return if $self->session->loaded; $self->session->load(); }
sub current_user { my $self = shift; if (@_) { my $currentuser_obj = shift; $self->session->set( 'user_id' => $currentuser_obj ? $currentuser_obj->id : undef ); $self->{current_user} = ( $currentuser_obj || undef ); } my $object; if ( defined $self->temporary_current_user ) { return $self->temporary_current_user; } elsif ( defined $self->{current_user} ) { return $self->{current_user}; } elsif ( my $id = $self->session->get('user_id') ) { $object = Jifty->app_class({require => 0}, "CurrentUser")->new( id => $id ); } elsif ( Jifty->admin_mode ) { $object = Jifty->app_class({require => 0}, "CurrentUser")->superuser; } else { $object = Jifty->app_class({require => 0}, "CurrentUser")->new; } # Don't cache the result unless the session had actually been # loaded already. $self->{current_user} = $object if $self->session->loaded; return $object; }
sub handle_request { my $self = shift; die _("No request to handle") unless Jifty->web->request; my ( $valid_actions, $denied_actions ) = $self->_validate_request_actions(); # In the case where we have a continuation and want to redirect if ( $self->request->continuation_path && Jifty->web->request->argument('_webservice_redirect') ) { # for continuation - perform internal redirect under webservices $self->webservices_redirect($self->request->continuation_path); return; } $self->request->save_continuation; unless ( $self->request->just_validating ) { $self->_process_valid_actions($valid_actions); $self->_process_denied_actions($denied_actions); } # If there's a continuation call, don't do the rest of this return if $self->response->success and $self->request->call_continuation; $self->redirect if $self->redirect_required; $self->request->do_mapping; } sub _process_denied_actions { my $self = shift; my $denied_actions = shift; for my $request_action (@$denied_actions) { my $action = $self->new_action_from_request($request_action); $action->deny( "Access Denied for " . ref($action) ); $self->response->result( $action->moniker => $action->result ); } } sub _validate_request_actions { my $self = shift; my @valid_actions; my @denied_actions; for my $request_action ( $self->request->actions ) { $self->log->debug( "Found action " . $request_action->class . " " . $request_action->moniker ); next unless $request_action->active; next if $request_action->has_run; unless ( $self->request->just_validating ) { unless ( Jifty->api->is_allowed( $request_action->class ) ) { $self->log->warn( "Attempt to call denied action '" . $request_action->class . "'" ); $self->log->warn( Jifty->api->explain($request_action->class ) ); # Possible cross-site request forgery $self->log->error("Action " . $request_action->class . " has been denied because the request is a GET") if $self->request->method eq "GET"; push @denied_actions, $request_action; next; } } # Make sure we can instantiate the action my $action = $self->new_action_from_request($request_action); next unless $action; $request_action->modified(0); # Try validating -- note that this is just the first pass; as # actions are run, they may fill in values which alter # validation of later actions $self->log->debug( "Validating action " . ref($action) . " " . $action->moniker ); $self->response->result( $action->moniker => $action->result ); $action->validate; push @valid_actions, $request_action; } return (\@valid_actions, \@denied_actions); } sub _process_valid_actions { my $self = shift; my $valid_actions = shift; for my $request_action (@$valid_actions) { # Pull the action out of the request (again, since # mappings may have affected parameters). This # returns the cached version unless the request has # been changed by argument mapping from previous # actions (Jifty::Request::Mapper) my $action = $self->new_action_from_request($request_action); next unless $action; if ( $request_action->modified ) { # If the request's action was changed, re-validate $action->result( Jifty::Result->new ); $action->result->action_class( ref $action ); $self->response->result( $action->moniker => $action->result ); $self->log->debug( "Re-validating action " . ref($action) . " " . $action->moniker ); next unless $action->validate; } $self->log->debug( "Running action " . ref($action) . " " . $action->moniker ); eval { $action->run; }; $request_action->has_run(1); if ( my $err = $@ ) { # Poor man's exception propagation; we need to get # "LAST RULE" and "ABORT" exceptions back up to the # dispatcher. This is specifically for redirects from # actions die $err if ( $err =~ /^(LAST RULE|ABORT)/ ); $self->log->fatal($err); $action->result->error( Jifty->config->framework("DevelMode") ? $err : _("There was an error completing the request. Please try again later." ) ); } # Fill in the request with any results that that action # may have yielded. $self->request->do_mapping; } }
sub form { my $self = shift; $self->{form} ||= Jifty::Web::Form->new; return $self->{form}; }
sub new_action { my $self = shift; # Handle new_action('CreateFoo', moniker => 'create_foo') unshift @_, 'class' if @_ % 2; my %args = ( class => undef, moniker => undef, arguments => {}, current_user => $self->current_user, @_ ); # "Untaint" -- the implementation class is provided by the client!) # Allows anything that a normal package name allows my $class = delete $args{class}; unless ( $class =~ /^([0-9a-zA-Z_:]+)$/ ) { $self->log->error( "Bad action implementation class name: ", $class ); return; } $class = $1; # 'untaint' # Prepend the base path (probably "App::Action") unless it's there already $class = Jifty->api->qualify($class); # The implementation class is provided by the client, so this # isn't a "shouldn't happen" return unless Jifty::Util->require( $class ); my $action; # XXX TODO bullet proof eval { $action = $class->new(%args) }; if ($@) { my $err = $@; $self->log->fatal($err); return; } $self->{'actions'}{ $action->moniker } = $action; return $action; }
sub new_action_from_request { my $self = shift; my $req_action = shift; return $self->{'actions'}{ $req_action->moniker } if $self->{'actions'}{ $req_action->moniker } and not $req_action->modified; $self->new_action( class => $req_action->class, moniker => $req_action->moniker, order => $req_action->order, arguments => $req_action->arguments || {} ); }
sub failed_actions { my $self = shift; my @actions; for my $req_action ($self->request->actions) { next unless $self->response->result($req_action->moniker); next unless $self->response->result($req_action->moniker)->failure; push @actions, $self->new_action_from_request($req_action); } return @actions; }
sub succeeded_actions { my $self = shift; my @actions; for my $req_action ($self->request->actions) { next unless $self->response->result($req_action->moniker); next unless $self->response->result($req_action->moniker)->success; push @actions, $self->new_action_from_request($req_action); } return @actions; }
sub redirect_required { my $self = shift; return ( 1 ) if $self->force_redirect; if (!$self->request->is_subrequest and $self->next_page and ( ( $self->next_page ne $self->request->path ) or $self->request->state_variables or $self->state_variables ) ) { return (1); } else { return undef; } }
sub webservices_redirect { my ( $self, $to ) = @_; # XXX: move to singlepage plugin my ($spa) = Jifty->find_plugin('Jifty::Plugin::SinglePage') or return; return if $self->failed_actions; Jifty->web->request->remove_state_variable( 'region-'.$spa->region_name ); Jifty->web->request->add_fragment( name => $spa->region_name, path => $to, arguments => {}, wrapper => 0 ); }
sub redirect { my $self = shift; my $redir_to = shift || $self->next_page || $self->request->path; my $page; if ( ref $redir_to and $redir_to->isa("Jifty::Web::Form::Clickable")) { $page = $redir_to; } else { $page = Jifty::Web::Form::Clickable->new(); #We set this after creation to ensure that plugins that massage clickables don't impact us $page->url($redir_to ); } carp "Don't include GET parameters in the redirect URL -- use a Jifty::Web::Form::Clickable instead. See L<Jifty::Web/redirect>" if $page->url =~ /\?/; my %overrides = ( @_ ); $page->parameter($_ => $overrides{$_}) for keys %overrides; my @actions = Jifty->web->request->actions; # To submit a Jifty::Action::Redirect, we don't need to serialize a continuation, # unlike any other kind of actions. my $redirect_to_url = '' ; if ( (grep { not $_->action_class->isa('Jifty::Action::Redirect') } values %{ { $self->response->results } }) or $self->request->state_variables or $self->state_variables or $self->request->continuation or grep { $_->active and not $_->class->isa('Jifty::Action::Redirect') } @actions ) { my $request = Jifty::Request->new(); $request->add_state_variable( key => $_->key, value => $_->value ) for $self->request->state_variables; $request->add_state_variable( key => $_, value => $self->{state_variables}->{$_} ) for keys %{ $self->{state_variables} }; for (@actions) { my $new_action = $request->add_action( moniker => $_->moniker, class => $_->class, order => $_->order, active => $_->active && (not $_->has_run), has_run => $_->has_run, arguments => $_->arguments, ); # Clear out filehandles, which don't go thorugh continuations well for (keys %{$new_action->arguments || {}}) { $new_action->arguments->{$_} = '' if ref($new_action->arguments->{$_}) eq "Fh" || ref($new_action->arguments->{$_}) eq "Jifty::Web::FileUpload"; } } my %parameters = ($page->parameters); $request->argument($_ => $parameters{$_}) for keys %parameters; # Apache, lighttpd, and HSS all do one pass of unescaping on # PATH_INFO, which is what $request->path is normally set to. # We should replicate that here. $request->path( URI::Escape::uri_unescape( $page->url ) ); $request->request_uri( URI->new($page->url)->path_query ); $request->method("GET"); # ..effectively. $request->scheme($self->request->scheme); $request->continuation($self->request->continuation); my $cont = Jifty::Continuation->new( request => $request, response => $self->response, parent => $self->request->continuation, ); $redirect_to_url = $page->url."?J:RETURN=" . $cont->id; } else { $redirect_to_url = $page->complete_url; } $self->_redirect($redirect_to_url); } sub _redirect { my $self = shift; my ($page) = @_; # It's an experimental feature to support redirect within a # region. if ($self->current_region) { # If we're within a region stack, we don't really want to # redirect. We want to redispatch. Also reset the things # applied on beofre. local $self->{navigation} = undef; local $self->{page_navigation} = undef; $self->replace_current_region($page); Jifty::Dispatcher::_abort(); return; } if (my $redir = Jifty->web->request->argument('_webservice_redirect')) { push @$redir, $page; return; } # $page can't lead with // or it assumes it's a URI scheme. $page =~ s{^/+}{/}; # This is designed to work under CGI or FastCGI; will need an # abstraction for mod_perl # Clear out the output, if any Jifty->handler->buffer->clear; my $response = Jifty->web->response; $self->log->debug("Execing redirect to $page"); # Headers.. $response->header( Location => $page ); $response->status( 302 ); # cookie has to be sent or returning from continuations breaks Jifty->web->session->set_cookie; # Mason abort, or dispatcher abort out of here $self->mason->abort if $self->mason; Jifty::Dispatcher::_abort(); }
sub caller { my $self = shift; return Jifty::Request->new unless $self->request->continuation; return $self->request->continuation->request; }
sub tangent { my $self = shift; if (@_ == 1 ) { $self->log->error("Jifty::Web->tangent takes a paramhash. Perhaps you passed '".$_[0]."' , rather than 'url => ".$_[0]."'"); die; } my $clickable = Jifty::Web::Form::Clickable->new( returns => { }, preserve_state => 1, @_ ); if ( defined wantarray ) { return $clickable->generate; } else { my $request = Jifty->web->request->clone; my %clickable = $clickable->get_parameters; $request->argument($_ => $clickable{$_}) for keys %clickable; local Jifty->web->{request} = $request; Jifty->web->request->save_continuation; } }
sub goto { my $self = shift; Jifty->web->redirect( Jifty::Web::Form::Clickable->new(@_)); }
sub link { my $self = shift; my $link = Jifty::Web::Form::Clickable->new(@_)->generate; return $link if defined wantarray; $link->render; }
sub return { my $self = shift; my %args = (to => undef, @_); my $continuation = Jifty->web->request->continuation; if (not $continuation and $args{to}) { $continuation = Jifty::Continuation->new( request => Jifty::Request->new(path => $args{to}) ); } delete $args{to}; my $clickable = Jifty::Web::Form::Clickable->new( call => $continuation, %args ); if ( defined wantarray ) { return $clickable->generate; } else { $self->redirect($clickable); } }
sub template_exists { my $self = shift; my $template = shift; my $value = ref $template ? $$template : $template; # Strip trailing slashes $value =~ s{/$}{} if $value ne "/"; foreach my $handler ( map {Jifty->handler->view($_)} Jifty->handler->view_handlers ) { if ( my $path = $handler->template_exists($value) ) { $$template = $path if ref $template; return $handler; } } return undef; } my %TEMPLATE_CACHE;
sub render_template { my $self = shift; my $template = shift; my $handler; my $content; my $void_context = ( defined wantarray ? 0 :1); # Check for ../../../../../etc/passwd my $abs_template_path = Jifty::Util->absolute_path( Jifty->config->framework('Web')->{'TemplateRoot'} . $template ); my $abs_root_path = Jifty::Util->absolute_path( Jifty->config->framework('Web')->{'TemplateRoot'} ); $template = "/errors/500" if $abs_template_path !~ /^\Q$abs_root_path\E/; # Look for a possible handler, and cache it for future requests. # With DevelMode, always look it up. if ( not exists $TEMPLATE_CACHE{$template} or Jifty->config->framework('DevelMode')) { my $found = $template; $handler = $self->template_exists( \$found ); # We don't cache failing URLs, so clients' can't cause us to # chew up memory by requesting 404's $TEMPLATE_CACHE{$template} = [ $found, $handler ] if $handler; } # Dig out the actual template (which could have a "/index.html" on # it, or what have you) and its handler. ($template, $handler) = @{$TEMPLATE_CACHE{$template} || [$template, undef] }; # Handle 404's unless ($handler) { return $self->render_template("/errors/404") unless defined $template and $template eq "/errors/404"; $self->log->warn("Can't find 404 page!"); die "ABORT"; } $self->log->debug("Showing path $template using @{[ref $handler]}"); my $start_depth = Jifty->handler->buffer->depth; Jifty->handler->buffer->push( private => 1 ) unless $void_context; Jifty->handler->call_trigger("before_render_template", $handler, $template); eval { $handler->show($template) }; # Handle parse errors my $err = $@; $content = Jifty->handler->buffer->pop unless $void_context; Jifty->handler->call_trigger("after_render_template", $handler, $template, $content); if ( $err and not (eval { $err->isa('HTML::Mason::Exception::Abort') } or $err =~ /^ABORT/) ) { $self->log->fatal("View error: $err") if $err; if ($template eq '/errors/500') { $self->log->warn("Can't render internal_error: $err"); # XXX Built-in static "oh noes" page? die "ABORT"; } # XXX: This may leave a half-written tag open $err->template_stack; Jifty->handler->buffer->pop while Jifty->handler->buffer->depth > $start_depth; # Save the request away, and redirect to an error page Jifty->web->response->error($err); my $c = Jifty::Continuation->new( request => Jifty->web->request->top_request, response => Jifty->web->response, parent => Jifty->web->request->continuation, ); # Redirect with a continuation Jifty->web->_redirect( "/errors/500?J:C=" . $c->id ); } elsif ($err) { Jifty->handler->buffer->pop while Jifty->handler->buffer->depth > $start_depth; die "ABORT"; } else { return $content; } }
sub render_messages { my $self = shift; my $only_moniker = ''; $only_moniker = shift if (@_); $self->render_error_messages($only_moniker); $self->render_success_messages($only_moniker); return ''; }
sub render_success_messages { my $self = shift; my $moniker = shift; $self->_render_messages('message', $moniker); return ''; }
sub render_error_messages { my $self = shift; my $moniker = shift; $self->_render_messages('error', $moniker); return ''; }
sub _render_messages { my $self = shift; my $type = shift; my $only_moniker = shift || ''; my %results = $self->response->results; %results = ($only_moniker => $results{$only_moniker}) if $only_moniker; return unless grep {$_->$type()} values %results; my $plural = $type . "s"; $self->out(qq{<div class="jifty results messages" id="$plural">}); foreach my $moniker ( sort keys %results ) { if ( my $text = $results{$moniker}->$type() ) { if ( ref $text eq 'ARRAY' ) { $text = join '', @$text; } elsif ( ref $text ) { $self->log->warn( ref($text) . " reference provided as result $type " . "for action $moniker (@{[$results{$moniker}->action_class]})" ); } $self->out( qq{<div class="$type $moniker">$text</div>} ); } } $self->out(qq{</div>}); }
sub query_string { my $self = shift; my %args = @_; my @params; while ( ( my $key, my $value ) = each %args ) { push @params, $key . "=" . $self->escape_uri( $value ); } return ( join( ';', @params ) ); }
sub escape { no warnings 'uninitialized'; my $self = shift; return join '', map {my $html = $_; Jifty::View::Mason::Handler::escape_utf8( \$html ); $html} @_; }
sub escape_uri { no warnings 'uninitialized'; my $self = shift; return join '', map {my $uri = $_; Jifty::View::Mason::Handler::escape_uri( \$uri ); $uri} @_; }
sub navigation { my $self = shift; if (!$self->{navigation}) { $self->{navigation} = Jifty::Web::Menu->new(); } return $self->{navigation}; }
sub page_navigation { my $self = shift; if (!$self->{page_navigation}) { $self->{page_navigation} = Jifty::Web::Menu->new(); } return $self->{page_navigation}; }
sub include_css { my $self = shift; # if there's no trigger, 0 is returned. if aborted/handled, undef # is returned. if ( defined $self->call_trigger( 'include_css', @_ )) { $self->out( qq[<link rel="stylesheet" type="text/css" href="/static/css/$_" />\n] ) for @{ Jifty->web->css_files }; } $self->call_trigger('after_include_css', @_); return ''; }
sub add_css { my $self = shift; Jifty->web->css_files([ @{ Jifty->web->css_files }, @_ ]); }
sub include_javascript { my $self = shift; for my $url ( @{ __PACKAGE__->external_javascript_libs } ) { $self->out( qq[<script type="text/javascript" src="$url"></script>\n] ); } # if there's no trigger, 0 is returned. if aborted/handled, undef # is returned. if ( defined $self->call_trigger('include_javascript', @_) ) { for my $file ( @{ __PACKAGE__->javascript_libs } ) { $self->out( qq[<script type="text/javascript" src="/static/js/$file"></script>\n] ); } } $self->call_trigger('after_include_javascript', @_); return ''; }
sub add_javascript { my $self = shift; Jifty->web->javascript_libs([ @{ Jifty->web->javascript_libs }, @_ ]); }
sub remove_javascript { my $self = shift; my %remove = map { $_ => 1 } @_; Jifty->web->javascript_libs([ grep { !$remove{$_} } @{ Jifty->web->javascript_libs } ]); }
sub add_external_javascript { my $self = shift; Jifty->web->external_javascript_libs([ @{ Jifty->web->external_javascript_libs }, @_ ]); }
sub get_variable { my $self = shift; my $name = shift; my $var = $self->request->state_variable($name); return undef unless ($var); return $var->value(); }
sub set_variable { my $self = shift; my $name = shift; my $value = shift; if (!defined($value)) { delete $self->{state_variables}{$name}; } else { $self->{state_variables}{$name} = $value; } }
sub state_variables { my $self = shift; return %{ $self->{state_variables} }; }
sub clear_state_variables { my $self = shift; $self->{state_variables} = {}; }
sub get_region { my $self = shift; my ($name) = @_; return $self->{'regions'}{$name}; }
sub region { my $self = shift; # Create a region my $region = Jifty::Web::PageRegion->new(@_) or return; # Render it $region->render; }
sub replace_current_region { my $self = shift; my $path = shift; return undef unless (my $region = $self->current_region); $region->force_path($path); $region->render; }
sub current_region { my $self = shift; return $self->{'region_stack'} ? $self->{'region_stack'}[-1] : undef; }
sub qualified_region { my $self = shift; join "-", map { ref($_) ? $_->name : $_ } @{ $self->{'region_stack'} || [] }, @_; }
sub qualified_parent_region { my $self = shift; my @region_stack = @{ $self->{'region_stack'} || [] }; pop @region_stack; join "-", map { ref($_) ? $_->name : $_ } @region_stack, @_; }
sub is_ssl { Jifty->web->request && Jifty->web->request->secure } 1;