<html>
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Backbone.Declarative.Views 4.1.1, Marionette demo with precompiled templates (AMD)</title>
<meta name="description" content="Backbone.Declarative.Views 4.1.1, Marionette demo with precompiled templates (AMD)">
<!-- ATTN Foundation loaded without JS - stay clear of JS-driven widgets, or add the JS setup -->
<!--
ATTN jQuery 1.x is used here, making the demo compatible with oldIE. As a side effect,
Foundation 5 JS can't be used reliably (requires jQuery 2.x). Stay clear, or switch to
jQuery 2.x, available in the top-level bower_components.
-->
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/foundation/5.5.3/css/normalize.css">
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/foundation/5.5.3/css/foundation.css">
<link rel="stylesheet" href="https://cdn.rawgit.com/hashchange/backbone.declarative.views/4.1.1/demo/amd/amd.css">
<!-- Fix HTML5 tags in oldIE and other lacking legacy browsers -->
<script src="https://cdnjs.cloudflare.com/ajax/libs/modernizr/2.8.3/modernizr.js"></script>
</head>
<body>
<section id="header">
<h1 class="row">AMD setup: Marionette, with precompiled templates</h1>
</section>
<section class="stats">
<h2 class="row">Performance statistics</h2>
</section>
<section class="content">
<h2 class="row">Content</h2>
<div class="container row"></div>
</section>
<script src="https://cdnjs.cloudflare.com/ajax/libs/require.js/2.3.1/require.js"></script>
<script src="https://cdn.rawgit.com/hashchange/backbone.declarative.views/4.1.1/demo/amd/jsbin/map-config.js"></script>
<script src="https://cdn.rawgit.com/hashchange/backbone.declarative.views/4.1.1/demo/amd/rjs/output/parts/vendor.js"></script>
</body>
</html>
// base.js
define( 'local.base',[
'underscore',
'backbone',
'usertiming',
'backbone.declarative.views'
], function ( _, Backbone, performance ) {
var eventBus = _.extend( {}, Backbone.Events ),
Model = Backbone.Model.extend(),
Collection = Backbone.Collection.extend(
{ model: Model },
{
create: function ( modelCount ) {
var i,
collection = new Collection(),
models = [];
for ( i = 0; i < modelCount; i++ ) models.push( new collection.model( { number: i + 1 } ) );
collection.reset( models );
return collection;
}
}
),
StatsView = Backbone.View.extend( {
template: "#stats-template",
parent: ".stats",
initialize: function ( options ) {
var compiledTemplate = this.declarativeViews.getCachedTemplate().compiled;
this.template = compiledTemplate || _.template( this.declarativeViews.getCachedTemplate(). html );
options || ( options = {} );
if ( options.parent ) this.parent = options.parent;
this.$parent = Backbone.$( this.parent );
this.$el.appendTo( this.$parent );
this.listenTo( eventBus, "createStats", this.render );
},
render: function ( stats ) {
stats.totalDuration = Math.round( this.getTotalDuration() );
this.$el.html( this.template( stats ) );
},
getTotalDuration: function () {
// Query the document height. This is to assess the true total render time, including the painting.
// The calculation of the document height should be blocked by the browser until all item elements have
// been painted.
var docHeight = $( document ).height();
performance.measure( "paint", "create-itemViews-start" );
return performance.getEntriesByName( "paint" )[0].duration;
}
} );
Backbone.DeclarativeViews.custom.compiler = _.template;
return {
Model: Model,
Collection: Collection,
StatsView: StatsView,
eventBus: eventBus
}
} );
// views-marionette.js
define( 'local.views-marionette',[
'underscore',
'backbone',
'marionette',
'usertiming',
'local.base',
'backbone.declarative.views'
], function ( _, Backbone, Marionette, performance, base ) {
// Integrate with Marionette
Backbone.DeclarativeViews.joinMarionette();
var MarionetteBaseView = Marionette.ItemView || Marionette.View, // Base view type, depending on Marionette version
ItemView = MarionetteBaseView.extend( {
appendTo: function ( $parent ) {
if ( !( $parent instanceof Backbone.$ ) ) $parent = Backbone.$( $parent );
$parent.append( this.$el );
return this;
}
} ),
ListView = Marionette.CollectionView.extend( {
initialize: function ( options ) {
options || ( options = {} ); // jshint ignore:line
if ( options.parent ) this.parent = options.parent;
this.$parent = Backbone.$( this.parent );
},
onBeforeRender: function () {
// Start timer
performance.clearMarks();
performance.clearMeasures();
performance.mark( "create-itemViews-start" );
},
onRender: function () {
var duration;
// Measure the time it took to create the itemViews
performance.measure( "create-itemViews", "create-itemViews-start" );
duration = performance.getEntriesByName( "create-itemViews" )[0].duration;
if ( ! this.$el.parent().length ) this.$el.appendTo( this.$parent );
base.eventBus.trigger( "createStats", { itemViewCount : this.children.length, duration: Math.round( duration ) } );
}
} );
return {
ItemView: ItemView,
ListView: ListView
}
} );
define('local.precompiled.templates',['handlebars.runtime'], function(Handlebars) {
Handlebars = Handlebars["default"]; var template = Handlebars.template, templates = Handlebars.templates = Handlebars.templates || {};
templates['item-template'] = template({"compiler":[7,">= 4.0.0"],"main":function(container,depth0,helpers,partials,data) {
var helper;
return "<!-- data-tag-name=\"li\" data-class-name=\"item\" -->\n"
+ container.escapeExpression(((helper = (helper = helpers.number || (depth0 != null ? depth0.number : depth0)) != null ? helper : helpers.helperMissing),(typeof helper === "function" ? helper.call(depth0 != null ? depth0 : {},{"name":"number","hash":{},"data":data}) : helper)))
+ "\n";
},"useData":true});
templates['list-template'] = template({"compiler":[7,">= 4.0.0"],"main":function(container,depth0,helpers,partials,data) {
return "<!--\n data-tag-name=\"ul\"\n data-class-name=\"list small-block-grid-4 medium-block-grid-8 large-block-grid-10\"\n no other content - template just exists for the el definition\n-->\n";
},"useData":true});
templates['stats-template'] = template({"compiler":[7,">= 4.0.0"],"main":function(container,depth0,helpers,partials,data) {
var helper, alias1=depth0 != null ? depth0 : {}, alias2=helpers.helperMissing, alias3="function", alias4=container.escapeExpression;
return "<!-- data-tag-name=\"p\" data-class-name=\"row\" -->\nGenerating "
+ alias4(((helper = (helper = helpers.itemViewCount || (depth0 != null ? depth0.itemViewCount : depth0)) != null ? helper : alias2),(typeof helper === alias3 ? helper.call(alias1,{"name":"itemViewCount","hash":{},"data":data}) : helper)))
+ " item views took "
+ alias4(((helper = (helper = helpers.duration || (depth0 != null ? depth0.duration : depth0)) != null ? helper : alias2),(typeof helper === alias3 ? helper.call(alias1,{"name":"duration","hash":{},"data":data}) : helper)))
+ "ms. In total, it took "
+ alias4(((helper = (helper = helpers.renderDuration || (depth0 != null ? depth0.renderDuration : depth0)) != null ? helper : alias2),(typeof helper === alias3 ? helper.call(alias1,{"name":"renderDuration","hash":{},"data":data}) : helper)))
+ "ms until they had all been appended to the DOM. Finally, when painting was over, "
+ alias4(((helper = (helper = helpers.totalDuration || (depth0 != null ? depth0.totalDuration : depth0)) != null ? helper : alias2),(typeof helper === alias3 ? helper.call(alias1,{"name":"totalDuration","hash":{},"data":data}) : helper)))
+ "ms had passed.\n";
},"useData":true});
return templates;
});
// marionette-precompiled.js
require( [
'local.base',
'local.views-marionette',
'marionette.handlebars',
'precompiled.declarative.handlebars.templates',
'local.precompiled.templates'
], function ( base, marionetteViews ) {
var count = 1000,
ItemView = marionetteViews.ItemView.extend( {
template: "item-template" // without leading "#": is ID of precompiled template, not selector
} ),
listView = new marionetteViews.ListView( {
template: "list-template", // without leading "#"
childView: ItemView,
parent: ".container",
collection: base.Collection.create( count )
} );
new base.StatsView( { template: "stats-template" } ); // without leading "#"
listView.render();
} );
define("local.marionette-precompiled", function(){});
Output
You can jump to the latest bin by adding /latest
to your URL
Keyboard Shortcuts
Shortcut | Action |
---|---|
ctrl + [num] | Toggle nth panel |
ctrl + 0 | Close focused panel |
ctrl + enter | Re-render output. If console visible: run JS in console |
Ctrl + l | Clear the console |
ctrl + / | Toggle comment on selected lines |
ctrl + ] | Indents selected lines |
ctrl + [ | Unindents selected lines |
tab | Code complete & Emmet expand |
ctrl + shift + L | Beautify code in active panel |
ctrl + s | Save & lock current Bin from further changes |
ctrl + shift + s | Open the share options |
ctrl + y | Archive Bin |
Complete list of JS Bin shortcuts |
JS Bin URLs
URL | Action |
---|---|
/ | Show the full rendered output. This content will update in real time as it's updated from the /edit url. |
/edit | Edit the current bin |
/watch | Follow a Code Casting session |
/embed | Create an embeddable version of the bin |
/latest | Load the very latest bin (/latest goes in place of the revision) |
/[username]/last | View the last edited bin for this user |
/[username]/last/edit | Edit the last edited bin for this user |
/[username]/last/watch | Follow the Code Casting session for the latest bin for this user |
/quiet | Remove analytics and edit button from rendered output |
.js | Load only the JavaScript for a bin |
.css | Load only the CSS for a bin |
Except for username prefixed urls, the url may start with http://jsbin.com/abc and the url fragments can be added to the url to view it differently. |