A View
is used for managing portions of the DOM via a single parent DOM element or el
.
It provides a consistent interface for managing the content of the el
which is typically
administered by serializing a Backbone.Model
or Backbone.Collection
and rendering
a template with the serialized data into the View
s el
.
The View
provides event delegation for capturing and handling DOM interactions as well as
the ability to separate concerns into smaller, managed child views.
View
includes:
- The DOM API
- Class Events
- DOM Interactions
- Child Event Bubbling
- Entity Events
- View Rendering
- Prerendered Content
- View Lifecycle
A View
can have Region
s and Behavior
s
- Instantiating a View
- Rendering a View
- View Lifecycle and Events
- Entity Events
- DOM Interactions
- Behaviors
- Managing Children
- Efficient Nested View Structures
- Listening to Events on Children
When instantiating a View
there are several properties, if passed,
that will be attached directly to the instance:
attributes
, behaviors
, childViewEventPrefix
, childViewEvents
,
childViewTriggers
, className
, collection
, collectionEvents
, el
,
events
, id
, model
, modelEvents
, regionClass
, regions
,
tagName
, template
, templateContext
, triggers
, ui
import { View } from 'backbone.marionette';
const myView = new View({ ... });
Some of these properties come from Marionette, but many are inherited from
Backbone.View
.
The Marionette View implements a powerful render method which, given a
template
, will build your
HTML from that template, mixing in model
or collection
data and any
extra template context.
Unlike Backbone.View
Marionette defines render
and this method should
not be overridden. To add functionality to the render use the
render
and before:render
events.
For more detail on how to render templates, see View Template Rendering.
By setting template
to false
you can entirely disable
the view rendering and events. This may be useful for cases where you only need the el
or have
prerendered content
that you do not intend to re-render.
An instantiated View
is aware of its lifecycle state and will throw events related to when that state changes.
The view states indicate whether the view is rendered, attached to the DOM, or destroyed.
Read More:
The View
can bind to events that occur on the attached model
and collection
- this
includes both standard backbone-events and custom events.
Read More:
In addition to what Backbone provides the views, Marionette has additional API
for DOM interactions: events
, triggers
, and ui
.
Read More:
A Behavior
provides a clean separation of concerns to your view logic,
allowing you to share common user-facing operations between your views.
Read More:
View
provides a simple interface for managing child-views with
showChildView
, getChildView
, and
detachChildView
.
These methods all access regions
within the view.
We will cover this here but for more advanced information, see the
documentation for regions.
The Marionette.View
class lets us manage a hierarchy of views using regions
.
Regions are a hook point that lets us show views inside views, manage the
show/hide lifecycles, and act on events inside the children.
This Section only covers the basics. For more information on regions, see the Regions Documentation.
Regions are ideal for rendering application layouts by isolating concerns inside another view. This is especially useful for independently re-rendering chunks of your application without having to completely re-draw the entire screen every time some data is updated.
Regions can be added to a View at class definition, with regions
,
or at runtime using addRegion
.
When you extend View
, we use the regions
attribute to point to the selector
where the new view will be displayed:
import _ from 'underscore';
import { View } from 'backbone.marionette';
const MyView = View.extend({
template: _.template(`
<div id="first-region"></div>
<div id="second-region"></div>
<div id="third-region"></div>
`),
regions: {
firstRegion: '#first-region',
secondRegion: '#second-region'
}
});
When we show views in the region, the contents of #first-region
and
#second-region
will be replaced with the contents of the view we show. The
value in the regions
hash is just a jQuery selector, and any valid jQuery
syntax will suffice.
To show a view inside a region, simply call showChildView(regionName, view)
. This
will handle rendering the view's HTML and attaching it to the DOM for you:
import _ from 'underscore';
import { View } from 'backbone.marionette';
import SubView from './subview';
const MyView = View.extend({
template: _.template('<h1>Title</h1><div id="first-region"></div>'),
regions: {
firstRegion: '#first-region'
},
onRender() {
this.showChildView('firstRegion', new SubView());
}
});
Note: If view.showChildView(region, subView)
is invoked before the view
has been rendered, it will automatically render the view
so the region's el
exists in the DOM.
To access the child view of a View
- use the getChildView(regionName)
method.
This will return the view instance that is currently being displayed at that
region, or null
:
import _ from 'underscore';
import { View } from 'backbone.marionette'
import SubView from './subview';
const MyView = View.extend({
template: _.template('<h1>Title</h1><div id="first-region"></div>'),
regions: {
firstRegion: '#first-region'
},
onRender() {
this.showChildView('firstRegion', new SubView());
},
onSomeEvent() {
const first = this.getChildView('firstRegion');
first.doSomething();
}
});
If no view is available, getChildView
returns null
.
You can detach a child view from a region through detachChildView(region)
import _ from 'underscore';
import { View } from 'backbone.marionette'
import SubView from './subview';
const MyView = View.extend({
template: _.template(`
<h1>Title</h1>
<div id="first-region"></div>
<div id="second-region"></div>
`),
regions: {
firstRegion: '#first-region',
secondRegion: '#second-region'
},
onRender() {
this.showChildView('firstRegion', new SubView());
},
onMoveView() {
const view = this.detachChildView('firstRegion');
this.showChildView('secondRegion', view);
}
});
This is a proxy for region.detachView()
There are two ways to easily destroy a child view.
// Directly
myChildView.getChildView('regionName').destroy();
// Indirectly
myChildView.getRegion('regionName').empty();
Any defined regions within a View
will be available to the View
or any
calling code immediately after rendering the View
. Using getRegion
or any
of the child view methods above will first render the view so that the region is
available.
When your views get some more regions, you may want to think of the most efficient way to render your views. Since manipulating the DOM is performance heavy, it's best practice to render most of your views at once.
Marionette provides a simple mechanism to infinitely nest views in a single
paint: just render all of the children in the onRender
callback for the
render
event.
import { View } from 'backbone.marionette';
const ParentView = View.extend({
// ...
onRender() {
this.showChildView('header', new HeaderView());
this.showChildView('footer', new FooterView());
}
});
myRegion.show(new ParentView());
In this example, the doubly-nested view structure will be rendered in a single paint.
This system is recursive, so it works for any deeply nested structure. The child views you show can render their own child views within their onRender callbacks!
Using regions lets you listen to the events that fire on child views - views attached inside a region. This lets a parent view take action depending on what events are triggered in views it directly owns.
Read More: