# @title Views
# Views
Views are the last step in the process of a request to a MVC framework such as
Ramaze. A controller loads a model, the model processes a certain amount of
data and the controller will then pass this data to a view. The typical flow of
a MVC framework (or any framework using a view system) looks like the following:
Request --> Controller --> View
^ |
| '--> Model
| |
'--<-----'
The contents of a view is what the visitor of a page will eventually see.
Looking back at the waiter example introduced in the :doc:`controllers` chapter
a view can be seen as our dinner. It's the end result we requested for but it
has been modified according to our order.
Ramaze has support for many different template engines and thus views can look
quite different. By default Ramaze uses a simple template engine called "Etanni",
Etanni was developed by Michael Fellinger exclusively for Ramaze and is a very
fast template engine. In this chapter we'll use Etanni to make it a bit easier
to understand how views work.
## Layouts
Ramaze also has a concept of layouts. Layouts are basically views for views and
are placed in the "layout" directory of your application. These views can be
used to display generic elements such as a navigation menu for all views.
Because of this it's recommended to only place action specific markup in your
views. Global elements such as navigation menus or page titles should go in a
layout.
In order to render a view inside a layout all you have to do is calling the
"content" instance variable. If we were to use Etanni (more on this later) our
code would look like the following:
#{@content}
In order to use a layout we have to tell Ramaze to do so in our controller.
Setting a layout can be done in a few different ways. The easiest way is using
the method "layout" in your controller as following:
class Blogs < Ramaze::Controller
layout 'default'
end
This will tell Ramaze to use the layout "default.xhtml" for the Blogs controller.
While suited for most people there comes a time when you're in the need of a
more dynamic way of assigning a layout. This can be done in two different ways.
The first way is passing a block to the layout() method:
class Blogs < Ramaze::Controller
layout do |path|
if path === 'index'
:index_layout
else
:alternative_layout
end
end
end
In this example two layouts are used, `index_layout` for the index() method and
`alternative_layout` for all other methods.
The second way is using the method `set_layout`. This method works almost
identical to layout() but has one big advantage: it can set method specific
layouts. Changing the code posted above so that it uses this method would look
like the following:
class Blogs < Ramaze::Controller
# Set our default layout
layout 'alternative_layout'
# Set our layout for the index() method
set_layout 'index_layout' => [:index]
end
The `set_layout` method takes a hash where the keys are the names of the layouts
to use and the values the methods for which to use each layout. Below is another
example that specifies a few extra layout/method combinations.
class Blogs < Ramaze::Controller
# Set our default layout
layout 'default'
# Set our layout for the index() method
set_layout 'main' => [:index, :edit], 'extra' => [:add, :synchronize]
end
Note: layouts should be set outside
controller actions. Doing so can lead to unexpected behaviour as the
layout won't be visible until the next request.
## Loading Views
Loading views can be done in two different ways. When not explicitly calling a
certain view (or returning any data from the controller) Ramaze will try to load
a matching view for the current action. If the method "edit" was called Ramaze
will try to load a view called "edit". Manually sending a response back can be
done by returning a string from the controller method that is called. Take a
look at the example below.
class Blogs < Ramaze::Controller
map '/'
def index
end
def custom
"This is my custom response!"
end
def other
render_view :foobar
end
end
If the user were to visit /index Ramaze would try to load the view "index.xhtml"
(``.xhtml`` is the extension for Etanni templates) but when the user goes to
/custom he'll always see the message "This is my custom response!". This is
because Ramaze will use the return value of a controller method as the body for
the response that will be sent back to the visitor.
Let's take a look at the other() method in our controller. As you can see it
calls a method `render_view`. This method is used to render a different view
(in this case foobar.xhtml) but without calling it's action, once this is done
the contents of the view will be returned to the controller. When calling custom
views you can use any of the following methods:
* render\_view
* render\_partial
* render\_file
* render\_custom
* render\_full
### render\_view
As mentioned earlier this method is used to render a view without calling it's
action. This can be useful if you have several methods that share the same view
but feed it different data. The usage of this method is quite simple, it's first
argument is the name of the view to load (without the extension, Ramaze will
figure this out) and the second argument is a hash containing any additional
variables to send to the view (more on this later).
# Render "foo.xhtml"
render_view :foo
# Render "foo.xhtml" and send some extra data to it
render_view :foo, :name => "Ramaze"
### render\_partial
The `render_partial` method works similar to the `render_view` method but with
two differences:
1. This method *does* execute a matching action.
2. This method *does not* render a layout.
This makes the `render_partial` method useful for responses to Ajax calls that
don't need (or don't want to) load both the view and the layout. This method has
the same arguments as the `render_view` method.
# Render the view "partial.xhtml"
render_partial :partial
# Render the partial "partial.xhtml" and send some extra data to it
render_partial :partial, :name => "Ramaze"
### render\_file
There comes a time when you may want to render a file that's located somewhere
else. For this there is the `render_file()`` method. This method takes a path,
either relative or absolute to a file that should be rendered. This can be
useful if you have a cache directory located outside of your project directory
and you want to load a view from it.
The first argument of this method is a path to a file to render, the second
argument is a hash with variables just like the other render methods.
Optionally this method accepts a block that can be used to modify the current
action.
# Render a file located in /tmp
render_file '/tmp/view.xhtml'
# Render a file and send some extra data to it
render_file '/tmp/view.xhtml', :name => "Ramaze"
# Render a file and modify the action
render_file '/tmp/view.xhtml' do |action|
# Remove our layout
action.layout = nil
end
### render\_custom
The render_custom() method can be used to create your personal render method and
is actually used by methods such as the render_partial method. The syntax is the
same as the render_file() method except that instead of a full path it's first
argument should be the name of the action to render.
render_custom :index do |action|
# Remove the layout
action.layout = nil
# Render the view without calling a method
action.method = nil
end
### render\_full
Last but not least there's the render_full() method. This is the method Ramaze
uses for calling a controller along with it's views and such. This method takes
two arguments, the first is the full path of the action to render and the second
a hash that will be used for the query string parameters. Please note that if this
method is called in the first request of a client you won't have access to the
session data and such, any following calls will have access to this data.
# Calls Blogs#index
render_full '/blog/index'
# Calls Blogs#edit(10)
render_full '/blog/edit/10'
### Assigning Data
Assigning data to a view is very simple. Ramaze will copy all instance variables
from the current controller into the view. This means that if you have a variable
@user set to "Yorick Peterse" this variable can be displayed in your view as
following (assuming you're using Etanni):
Username: #{@user}
Besides this you can assign custom data to a view by calling any of the render
methods and passing a hash to them.
Please note that the behavior or the syntax of displaying variables may change
depending on the template engine you're using. While Etanni allows you to execute
plain Ruby code in your view a template engine such as Mustache won't and thus
may have a different syntax. If we wanted to use Mustache and display our @user
variable it would have to be done as following:
Username: {{user}}
## View Mapping
Views are saved in the directory "view" and are saved according to the controller
mapping. If you have a controller that's mapped to /modules/blog the index view
will be located at view/modules/blog/index.xhtml. Below are a few examples that
show where the views are located when passing different values to the map()
method.
map '/' # view/index.html, view/edit.xhtml, etc
map '/blog' # view/blog/index.xhtml, view/blog/edit.xhtml, etc
map '/modules/blog' # view/modules/blog/index.xhtml, view/modules/blog/edit.xhtml, etc
## Template Engines
Ramaze ships with support for the following template engines:
* Erector
* Etanni
* Erubis
* Ezamar
* HAML
* Less
* Gestalt
* Liquid
* Lokar
* Mustache
* Nagoro
* Remarkably
* Sass
* Slippers
* Tagz
* Tenjin
All of these engines can be used on a per controller basis by calling the
engine() method and passing a symbol or string to it.
class Blogs < Ramaze::Controller
engine :etanni
end
The engine() method uses the provide() method (more on that in a second) to set
the given engine for all data sent back to the visitor with a content type of
"text/html". If you want to use a certain engine for a different content type
(e.g. application/json) you can do so using the provide() method:
class Blogs < Ramaze::Controller
# Simple right?
provide(:json, :erb)
# Somewhat more verbose
provide(:json, :engine => :erb)
# AWESOME!
provide(:json, :type => 'application/json') do |action, value|
# "value" is the response body from our controller's method
value.to_json
end
end
It's important to remember that the actions in the provide() method will only
be executed if it's first parameter (in this case "json") are appended to the
URL as an extension. A request to /hello would output HTML when using the above
code, if we wanted JSON we'd have to send a request to /hello.json.
The default template engine used by Ramaze is Etanni. Etanni is a small template
engine that ships with Ramaze (and Innate!) that's extremely fast and has a very
simple syntax. Statements are wrapped in tags and rendering data can be
done by wrapping the variables in ``#{}``:
# accepts regular Ruby code
Hello Yorick
And who are you?
# Display our name
#{@user.name}
Etanni can be very useful for most project but it's *not* recommended to use it
when you want to allow a client to manage certain templates (e.g. Email layouts).
This is because Etanni allows you to execute regular Ruby code and thus someone
could do some serious damage to your server without even knowing it.