私信发送成功
检测到您已登录技术前沿网,是否 一键登录码云
Watch Star Fork

demoeig / rackJavaScript

kesin 提交于   master
commit 893f2474cf5e59e442f4ef8dc526f9cfdd4fd69d
0 parents

33 文件发生了变化, 影响行数: +3842 -0

  • .DS_Store +0 -0
  • Gemfile +2 -0
  • Gemfile.lock +10 -0
  • README.md +140 -0
  • config.ru +14 -0
  • css/base.css +0 -0
  • index.html +39 -0
  • release/app-debug.js +7 -0
  • release/app.js +1 -0
  • release/toaster/app.js +18 -0
  • release/toaster/glue.js +33 -0
  • release/toaster/gui.js +69 -0
  • release/toaster/local_storage.js +48 -0
  • release/toaster/use_case.js +37 -0
  • release/toaster/utils.js +62 -0
  • src/app.coffee +17 -0
  • src/glue.coffee +12 -0
  • src/gui.coffee +30 -0
  • src/local_storage.coffee +17 -0
  • src/use_case.coffee +15 -0
  • src/utils.coffee +27 -0
  • toaster.coffee +19 -0
  • touch +0 -0
  • vendors/YouAreDaBomb.js +125 -0
  • vendors/YouAreDaChef.js +105 -0
  • vendors/base.js +7 -0
  • vendors/handlebars-1.0.0.beta.6.js +1550 -0
  • vendors/jquery-1.7.2.min.js +4 -0
  • vendors/jquery.json-2.3.min.js +23 -0
  • vendors/jstorage.min.js +8 -0
  • vendors/sugar-1.3.min.js +111 -0
  • vendors/underscore.js +1059 -0
  • vendors/uuid.js +233 -0

不支持预览此文件

@@ -0,0 +1,2 @@
+source :rubygems
+gem 'rack'
@@ -0,0 +1,10 @@
+GEM
+ remote: http://rubygems.org/
+ specs:
+ rack (1.5.2)
+
+PLATFORMS
+ ruby
+
+DEPENDENCIES
+ rack
@@ -0,0 +1,140 @@
+hexagonal.js hello world example
+====================
+
+This is the simplest possible hexagonaljs app.
+It shows all the basic concepts of a hexagonaljs architecture.
+
+
+How to run it?
+--------------
+
+Just clone the repo and open index.html.
+
+How to work on it?
+------------------
+
+ $ npm install -g coffee-toaster
+
+ $ toaster -dw
+
+
+The app object
+--------------
+
+This is the starting point of the application. The app object initializes the use case object and the adapters. At the end it passes all of them to the glue code. Often it's also starting the useCase object.
+
+```coffeescript
+#<< utils
+#<< local_storage
+#<< use_case
+#<< gui
+#<< glue
+
+class App
+ constructor: ->
+ useCase = new UseCase()
+ gui = new Gui()
+ glue = new Glue(useCase, gui)
+
+ useCase.start()
+
+new App()
+```
+
+The use case object
+-------------------
+
+At the heart of an application lives the use case object. Let's look at the use case in this project:
+
+```coffeescript
+class UseCase
+ constructor: ->
+
+ start: =>
+ @askForName()
+
+ askForName: =>
+
+ nameProvided: (name) =>
+ @greetUser(name)
+
+ greetUser: (name) =>
+
+ restart: =>
+ @askForName()
+```
+
+The goal of the use case is to show the most important domain actions the user (or other actors) can make. It should be possible to see what kind of interaction happens in the use case. There should be nothing about GUI here. There should be nothing about the persistence (AJAX/API calls) here.
+
+The gui object
+--------------
+
+The idea of hexagonal architecture is based on the idea of adapters that are plugged into the domain. One of typical adapters is the gui object. It's responsible for reacting to useCase actions and for letting the user interact with the app. The gui itself shouldn't have any logic. Often, it's just a proxy between the user and the app.
+
+```coffeescript
+class Gui
+ constructor: ->
+
+ createElementFor: (templateId, data) =>
+ source = $(templateId).html()
+ template = Handlebars.compile(source)
+ html = template(data)
+ element = $(html)
+
+ showAskForName: =>
+ element = @createElementFor("#ask-for-name-template")
+ $(".main").append(element)
+ confirmNameButton = $("#confirm-name-button")
+ confirmNameButton.click( => @confirmNameButtonClicked($("#name-input").val()))
+ $("#name-input").focus()
+
+ confirmNameButtonClicked: (name) =>
+
+ hideAskForName: =>
+ $(".ask-for-name").remove()
+
+ showGreetMessage: (name) =>
+ element = @createElementFor("#greet-message-template", {name : name})
+ $(".main").append(element)
+ $("#restart-link").click( => @restartClicked())
+
+ restartClicked: =>
+
+ hideGreetMessage: =>
+ $(".greet-message").remove()
+```
+
+In this example, jQuery is used for selectors and basic DOM manipulation (remove, append etc). Handlebars is used for HTML rendering. As you see, some of the methods here are the GUI equivalents of the use case object methods, like: askForName -> showAskForName, greetUser -> showGreetMessage.
+
+The glue object
+---------------
+
+The use case and gui objects don't know each other. It's great for reducing coupling and better testability. At some point, though, you need to run them together. We call it a glue object. It's responsible for merging the useCase and gui objects. In larger apps, it's also used for merging other adapters, like ServerSideAdapter, LocalStorageAdapter, PusherAdapter, SoundAdapter, etc.
+
+```coffeescript
+class Glue
+ constructor: (@useCase, @gui, @storage)->
+ After(@useCase, "askForName", => @gui.showAskForName())
+ After(@useCase, "nameProvided", => @gui.hideAskForName())
+ After(@useCase, "greetUser", (name) => @gui.showGreetMessage(name))
+ After(@useCase, "restart", => @gui.hideGreetMessage())
+
+ After(@gui, "restartClicked", => @useCase.restart())
+ After(@gui, "confirmNameButtonClicked", (name) => @useCase.nameProvided(name))
+```
+
+You can find the definition of the After method in the utils.coffee file. It's basically wrapping one function with another. If you look at:
+
+```coffeescript
+After(@useCase, "askForName", => @gui.showAskForName())
+```
+
+What it does, is replacing the original @useCase.askForName method with something like:
+
+```coffeescript
+askForName: =>
+ # original code
+ => @gui.showAskForName()
+```
+
+The use case and gui are merged runtime. The merge rules are defines in the glue code.
@@ -0,0 +1,14 @@
+use Rack::Static,
+ :urls => ["/release", "/vendors", "/css"],
+ :root => "."
+
+run lambda { |env|
+ [
+ 200,
+ {
+ 'Content-Type' => 'text/html',
+ 'Cache-Control' => 'public, max-age=86400'
+ },
+ File.open('index.html', File::RDONLY)
+ ]
+}
css/base.css 0 → 100644
查看文件 @ 893f247
@@ -0,0 +1,39 @@
+<!doctype html>
+<html lang="en">
+<head>
+<meta charset="utf-8">
+<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
+<title>hexagonal.js • starter app</title>
+<link rel="stylesheet" href="css/base.css">
+</head>
+<body>
+ <div class="main">
+ </div>
+
+ <script type="x-handlebars-template" id="ask-for-name-template">
+ <div class="ask-for-name">
+ <p>What is your name?</p>
+ <input type="text" id="name-input"/>
+ <input type="button" id="confirm-name-button" value="Confirm"/>
+ </div>
+ </script>
+
+ <script type="x-handlebars-template" id="greet-message-template">
+ <div class="greet-message">
+ <p>Hello, {{name}}</p>
+ <a href="#" id="restart-link">Restart</a>
+ </div>
+ </script>
+
+ <script src="vendors/base.js"></script>
+ <script src="vendors/jquery-1.7.2.min.js" type="text/javascript"></script>
+ <script src="vendors/jquery.json-2.3.min.js" type="text/javascript"></script>
+ <script src="vendors/uuid.js" type="text/javascript"></script>
+ <script src="vendors/jstorage.min.js" type="text/javascript"></script>
+ <script src="vendors/underscore.js" type="text/javascript"></script>
+ <script src="vendors/sugar-1.3.min.js" type="text/javascript"></script>
+ <script src="vendors/handlebars-1.0.0.beta.6.js" type="text/javascript"></script>
+ <script src="vendors/YouAreDaBomb.js" type="text/javascript"></script>
+ <script src="release/app-debug.js" type="text/javascript"></script>
+</body>
+</html>
@@ -0,0 +1,7 @@
+
+document.write('<scri'+'pt src="release/toaster/utils.js"></scr'+'ipt>')
+document.write('<scri'+'pt src="release/toaster/local_storage.js"></scr'+'ipt>')
+document.write('<scri'+'pt src="release/toaster/use_case.js"></scr'+'ipt>')
+document.write('<scri'+'pt src="release/toaster/gui.js"></scr'+'ipt>')
+document.write('<scri'+'pt src="release/toaster/glue.js"></scr'+'ipt>')
+document.write('<scri'+'pt src="release/toaster/app.js"></scr'+'ipt>')
\ No newline at end of file
@@ -0,0 +1 @@
+var App,Glue,Gui,LocalStorage,UseCase,__hasProp={}.hasOwnProperty,__bind=function(a,b){return function(){return a.apply(b,arguments)}};_.defaults(this,{Before:function(a,b,c){return YouAreDaBomb(a,b).before(c)},BeforeAnyCallback:function(a,b,c){return YouAreDaBomb(a,b).beforeAnyCallback(c)},After:function(a,b,c){return YouAreDaBomb(a,b).after(c)},Around:function(a,b,c){return YouAreDaBomb(a,b).around(c)},AfterAll:function(a,b,c){var d,e,f,g;g=[];for(e=0,f=b.length;e<f;e++)d=b[e],g.push(After(a,d,c));return g},LogAll:function(a){var b,c,d;d=[];for(b in a){if(!__hasProp.call(a,b))continue;c=a[b],_.isFunction(c)?d.push(function(b){return Before(a,b,function(){return console.log("calling: "+b)})}(b)):d.push(void 0)}return d},AutoBind:function(a,b){var c,d,e;e=[];for(c in a)d=a[c],_.isFunction(d)?e.push(function(c){if(c.endsWith("Clicked")&&b[c.remove("Clicked")])return After(a,c,function(a){return b[c.remove("Clicked")](a)})}(c)):e.push(void 0);return e}}),LocalStorage=function(){function a(a){this.namespace=a,this.flush=__bind(this.flush,this),this.remove=__bind(this.remove,this),this.get=__bind(this.get,this),this.set=__bind(this.set,this)}return a.prototype.set=function(a,b){return console.log(b),$.jStorage.set(""+this.namespace+"/"+a,b)},a.prototype.get=function(a){return $.jStorage.get(""+this.namespace+"/"+a)},a.prototype.remove=function(a){return $.jStorage.deleteKey(""+this.namespace+"/"+a)},a.prototype.flush=function(){var a,b,c,d,e;d=$.jStorage.index(),e=[];for(b=0,c=d.length;b<c;b++)a=d[b],a.match("^"+this.namespace)?e.push($.jStorage.deleteKey(a)):e.push(void 0);return e},a}(),UseCase=function(){function a(){this.restart=__bind(this.restart,this),this.greetUser=__bind(this.greetUser,this),this.nameProvided=__bind(this.nameProvided,this),this.askForName=__bind(this.askForName,this),this.start=__bind(this.start,this)}return a.prototype.start=function(){return this.askForName()},a.prototype.askForName=function(){},a.prototype.nameProvided=function(a){return this.greetUser(a)},a.prototype.greetUser=function(a){},a.prototype.restart=function(){return this.askForName()},a}(),Gui=function(){function a(){this.hideGreetMessage=__bind(this.hideGreetMessage,this),this.restartClicked=__bind(this.restartClicked,this),this.showGreetMessage=__bind(this.showGreetMessage,this),this.hideAskForName=__bind(this.hideAskForName,this),this.confirmNameButtonClicked=__bind(this.confirmNameButtonClicked,this),this.showAskForName=__bind(this.showAskForName,this),this.createElementFor=__bind(this.createElementFor,this)}return a.prototype.createElementFor=function(a,b){var c,d,e,f;return e=$(a).html(),f=Handlebars.compile(e),d=f(b),c=$(d)},a.prototype.showAskForName=function(){var a,b,c=this;return b=this.createElementFor("#ask-for-name-template"),$(".main").append(b),a=$("#confirm-name-button"),a.click(function(){return c.confirmNameButtonClicked($("#name-input").val())}),$("#name-input").focus()},a.prototype.confirmNameButtonClicked=function(a){},a.prototype.hideAskForName=function(){return $(".ask-for-name").remove()},a.prototype.showGreetMessage=function(a){var b,c=this;return b=this.createElementFor("#greet-message-template",{name:a}),$(".main").append(b),$("#restart-link").click(function(){return c.restartClicked()})},a.prototype.restartClicked=function(){},a.prototype.hideGreetMessage=function(){return $(".greet-message").remove()},a}(),Glue=function(){function a(a,b){var c=this;this.useCase=a,this.gui=b,After(this.useCase,"askForName",function(){return c.gui.showAskForName()}),After(this.useCase,"nameProvided",function(){return c.gui.hideAskForName()}),After(this.useCase,"greetUser",function(a){return c.gui.showGreetMessage(a)}),After(this.useCase,"restart",function(){return c.gui.hideGreetMessage()}),After(this.gui,"restartClicked",function(){return c.useCase.restart()}),After(this.gui,"confirmNameButtonClicked",function(a){return c.useCase.nameProvided(a)}),LogAll(this.useCase),LogAll(this.gui)}return a}(),App=function(){function a(){var a,b,c;c=new UseCase,b=new Gui,a=new Glue(c,b),c.start(),window.useCase=c}return a}(),new App
\ No newline at end of file
@@ -0,0 +1,18 @@
+var App;
+
+App = (function() {
+
+ function App() {
+ var glue, gui, useCase;
+ useCase = new UseCase();
+ gui = new Gui();
+ glue = new Glue(useCase, gui);
+ useCase.start();
+ window.useCase = useCase;
+ }
+
+ return App;
+
+})();
+
+new App();
@@ -0,0 +1,33 @@
+var Glue;
+
+Glue = (function() {
+
+ function Glue(useCase, gui) {
+ var _this = this;
+ this.useCase = useCase;
+ this.gui = gui;
+ After(this.useCase, "askForName", function() {
+ return _this.gui.showAskForName();
+ });
+ After(this.useCase, "nameProvided", function() {
+ return _this.gui.hideAskForName();
+ });
+ After(this.useCase, "greetUser", function(name) {
+ return _this.gui.showGreetMessage(name);
+ });
+ After(this.useCase, "restart", function() {
+ return _this.gui.hideGreetMessage();
+ });
+ After(this.gui, "restartClicked", function() {
+ return _this.useCase.restart();
+ });
+ After(this.gui, "confirmNameButtonClicked", function(name) {
+ return _this.useCase.nameProvided(name);
+ });
+ LogAll(this.useCase);
+ LogAll(this.gui);
+ }
+
+ return Glue;
+
+})();
@@ -0,0 +1,69 @@
+var Gui,
+ __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+
+Gui = (function() {
+
+ function Gui() {
+ this.hideGreetMessage = __bind(this.hideGreetMessage, this);
+
+ this.restartClicked = __bind(this.restartClicked, this);
+
+ this.showGreetMessage = __bind(this.showGreetMessage, this);
+
+ this.hideAskForName = __bind(this.hideAskForName, this);
+
+ this.confirmNameButtonClicked = __bind(this.confirmNameButtonClicked, this);
+
+ this.showAskForName = __bind(this.showAskForName, this);
+
+ this.createElementFor = __bind(this.createElementFor, this);
+
+ }
+
+ Gui.prototype.createElementFor = function(templateId, data) {
+ var element, html, source, template;
+ source = $(templateId).html();
+ template = Handlebars.compile(source);
+ html = template(data);
+ return element = $(html);
+ };
+
+ Gui.prototype.showAskForName = function() {
+ var confirmNameButton, element,
+ _this = this;
+ element = this.createElementFor("#ask-for-name-template");
+ $(".main").append(element);
+ confirmNameButton = $("#confirm-name-button");
+ confirmNameButton.click(function() {
+ return _this.confirmNameButtonClicked($("#name-input").val());
+ });
+ return $("#name-input").focus();
+ };
+
+ Gui.prototype.confirmNameButtonClicked = function(name) {};
+
+ Gui.prototype.hideAskForName = function() {
+ return $(".ask-for-name").remove();
+ };
+
+ Gui.prototype.showGreetMessage = function(name) {
+ var element,
+ _this = this;
+ element = this.createElementFor("#greet-message-template", {
+ name: name
+ });
+ $(".main").append(element);
+ return $("#restart-link").click(function() {
+ return _this.restartClicked();
+ });
+ };
+
+ Gui.prototype.restartClicked = function() {};
+
+ Gui.prototype.hideGreetMessage = function() {
+ return $(".greet-message").remove();
+ };
+
+ return Gui;
+
+})();
@@ -0,0 +1,48 @@
+var LocalStorage,
+ __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+
+LocalStorage = (function() {
+
+ function LocalStorage(namespace) {
+ this.namespace = namespace;
+ this.flush = __bind(this.flush, this);
+
+ this.remove = __bind(this.remove, this);
+
+ this.get = __bind(this.get, this);
+
+ this.set = __bind(this.set, this);
+
+ }
+
+ LocalStorage.prototype.set = function(key, value) {
+ console.log(value);
+ return $.jStorage.set("" + this.namespace + "/" + key, value);
+ };
+
+ LocalStorage.prototype.get = function(key) {
+ return $.jStorage.get("" + this.namespace + "/" + key);
+ };
+
+ LocalStorage.prototype.remove = function(key) {
+ return $.jStorage.deleteKey("" + this.namespace + "/" + key);
+ };
+
+ LocalStorage.prototype.flush = function() {
+ var key, _i, _len, _ref, _results;
+ _ref = $.jStorage.index();
+ _results = [];
+ for (_i = 0, _len = _ref.length; _i < _len; _i++) {
+ key = _ref[_i];
+ if (key.match("^" + this.namespace)) {
+ _results.push($.jStorage.deleteKey(key));
+ } else {
+ _results.push(void 0);
+ }
+ }
+ return _results;
+ };
+
+ return LocalStorage;
+
+})();
@@ -0,0 +1,37 @@
+var UseCase,
+ __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+
+UseCase = (function() {
+
+ function UseCase() {
+ this.restart = __bind(this.restart, this);
+
+ this.greetUser = __bind(this.greetUser, this);
+
+ this.nameProvided = __bind(this.nameProvided, this);
+
+ this.askForName = __bind(this.askForName, this);
+
+ this.start = __bind(this.start, this);
+
+ }
+
+ UseCase.prototype.start = function() {
+ return this.askForName();
+ };
+
+ UseCase.prototype.askForName = function() {};
+
+ UseCase.prototype.nameProvided = function(name) {
+ return this.greetUser(name);
+ };
+
+ UseCase.prototype.greetUser = function(name) {};
+
+ UseCase.prototype.restart = function() {
+ return this.askForName();
+ };
+
+ return UseCase;
+
+})();
@@ -0,0 +1,62 @@
+var __hasProp = {}.hasOwnProperty;
+
+_.defaults(this, {
+ Before: function(object, methodName, adviseMethod) {
+ return YouAreDaBomb(object, methodName).before(adviseMethod);
+ },
+ BeforeAnyCallback: function(object, methodName, adviseMethod) {
+ return YouAreDaBomb(object, methodName).beforeAnyCallback(adviseMethod);
+ },
+ After: function(object, methodName, adviseMethod) {
+ return YouAreDaBomb(object, methodName).after(adviseMethod);
+ },
+ Around: function(object, methodName, adviseMethod) {
+ return YouAreDaBomb(object, methodName).around(adviseMethod);
+ },
+ AfterAll: function(object, methodNames, adviseMethod) {
+ var methodName, _i, _len, _results;
+ _results = [];
+ for (_i = 0, _len = methodNames.length; _i < _len; _i++) {
+ methodName = methodNames[_i];
+ _results.push(After(object, methodName, adviseMethod));
+ }
+ return _results;
+ },
+ LogAll: function(object) {
+ var key, value, _results;
+ _results = [];
+ for (key in object) {
+ if (!__hasProp.call(object, key)) continue;
+ value = object[key];
+ if (_.isFunction(value)) {
+ _results.push((function(key) {
+ return Before(object, key, function() {
+ return console.log("calling: " + key);
+ });
+ })(key));
+ } else {
+ _results.push(void 0);
+ }
+ }
+ return _results;
+ },
+ AutoBind: function(gui, useCase) {
+ var key, value, _results;
+ _results = [];
+ for (key in gui) {
+ value = gui[key];
+ if (_.isFunction(value)) {
+ _results.push((function(key) {
+ if (key.endsWith("Clicked") && useCase[key.remove("Clicked")]) {
+ return After(gui, key, function(args) {
+ return useCase[key.remove("Clicked")](args);
+ });
+ }
+ })(key));
+ } else {
+ _results.push(void 0);
+ }
+ }
+ return _results;
+ }
+});
@@ -0,0 +1,17 @@
+#<< utils
+#<< local_storage
+#<< use_case
+#<< gui
+#<< glue
+
+class App
+ constructor: ->
+ useCase = new UseCase()
+ gui = new Gui()
+ glue = new Glue(useCase, gui)
+
+ useCase.start()
+ window.useCase = useCase
+
+new App()
+
@@ -0,0 +1,12 @@
+class Glue
+ constructor: (@useCase, @gui)->
+ After(@useCase, "askForName", => @gui.showAskForName())
+ After(@useCase, "nameProvided", => @gui.hideAskForName())
+ After(@useCase, "greetUser", (name) => @gui.showGreetMessage(name))
+ After(@useCase, "restart", => @gui.hideGreetMessage())
+
+ After(@gui, "restartClicked", => @useCase.restart())
+ After(@gui, "confirmNameButtonClicked", (name) => @useCase.nameProvided(name))
+
+ LogAll(@useCase)
+ LogAll(@gui)
\ No newline at end of file
@@ -0,0 +1,30 @@
+class Gui
+ constructor: ->
+
+ createElementFor: (templateId, data) =>
+ source = $(templateId).html()
+ template = Handlebars.compile(source)
+ html = template(data)
+ element = $(html)
+
+ showAskForName: =>
+ element = @createElementFor("#ask-for-name-template")
+ $(".main").append(element)
+ confirmNameButton = $("#confirm-name-button")
+ confirmNameButton.click( => @confirmNameButtonClicked($("#name-input").val()))
+ $("#name-input").focus()
+
+ confirmNameButtonClicked: (name) =>
+
+ hideAskForName: =>
+ $(".ask-for-name").remove()
+
+ showGreetMessage: (name) =>
+ element = @createElementFor("#greet-message-template", {name : name})
+ $(".main").append(element)
+ $("#restart-link").click( => @restartClicked())
+
+ restartClicked: =>
+
+ hideGreetMessage: =>
+ $(".greet-message").remove()
\ No newline at end of file
@@ -0,0 +1,17 @@
+class LocalStorage
+ constructor: (@namespace) ->
+
+ set: (key, value) =>
+ console.log(value)
+ $.jStorage.set("#{@namespace}/#{key}", value)
+
+ get: (key) =>
+ $.jStorage.get("#{@namespace}/#{key}")
+
+ remove: (key) =>
+ $.jStorage.deleteKey("#{@namespace}/#{key}")
+
+ flush: =>
+ for key in $.jStorage.index()
+ if key.match("^#{@namespace}")
+ $.jStorage.deleteKey(key)
@@ -0,0 +1,15 @@
+class UseCase
+ constructor: ->
+
+ start: =>
+ @askForName()
+
+ askForName: =>
+
+ nameProvided: (name) =>
+ @greetUser(name)
+
+ greetUser: (name) =>
+
+ restart: =>
+ @askForName()
\ No newline at end of file
@@ -0,0 +1,27 @@
+_.defaults this,
+ Before: (object, methodName, adviseMethod) ->
+ YouAreDaBomb(object, methodName).before(adviseMethod)
+ BeforeAnyCallback: (object, methodName, adviseMethod) ->
+ YouAreDaBomb(object, methodName).beforeAnyCallback(adviseMethod)
+ After: (object, methodName, adviseMethod) ->
+ YouAreDaBomb(object, methodName).after(adviseMethod)
+ Around: (object, methodName, adviseMethod) ->
+ YouAreDaBomb(object, methodName).around(adviseMethod)
+
+ AfterAll: (object, methodNames, adviseMethod) ->
+ for methodName in methodNames
+ After(object, methodName, adviseMethod)
+
+ LogAll: (object) ->
+ for own key, value of object
+ if _.isFunction(value)
+ do (key) ->
+ Before(object, key, -> console.log("calling: #{key}"))
+
+ AutoBind: (gui, useCase) ->
+ for key, value of gui
+ if _.isFunction(value)
+ do (key) ->
+ if key.endsWith("Clicked") and useCase[key.remove("Clicked")]
+ After(gui, key, (args) -> useCase[key.remove("Clicked")](args))
+
@@ -0,0 +1,19 @@
+# => SRC FOLDER
+toast 'src'
+
+ # EXCLUDED FOLDERS (optional)
+ # exclude: ['folder/to/exclude', 'another/folder/to/exclude', ... ]
+
+ # => VENDORS (optional)
+ # vendors: ['vendors/underscore.js']
+
+ # => OPTIONS (optional, default values listed)
+ bare: true
+ packaging: true
+ # expose: ''
+ # minify: true
+
+ # => HTTPFOLDER (optional), RELEASE / DEBUG (required)
+ httpfolder: 'release'
+ release: 'release/app.js'
+ debug: 'release/app-debug.js'
@@ -0,0 +1,125 @@
+(function() {
+ var __slice = Array.prototype.slice, __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+ _.defaults(this, {
+ YouAreDaBomb: __bind(function() {
+ var combinator, functionNames, object;
+ object = arguments[0], functionNames = 2 <= arguments.length ? __slice.call(arguments, 1) : [];
+ return combinator = {
+ before: __bind(function(advice) {
+ var functionName, _i, _len;
+ for (_i = 0, _len = functionNames.length; _i < _len; _i++) {
+ functionName = functionNames[_i];
+ object[functionName] = (__bind(function() {
+ var fn;
+ fn = object[functionName];
+ return function() {
+ var args, result;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ advice.apply(null, args);
+ result = fn.apply(object, args);
+ return result;
+ };
+ }, this))();
+ }
+ return combinator;
+ }, this),
+ around: function() {
+ var advice, functionName, pointcut_exprs, _i, _j, _len;
+ pointcut_exprs = 2 <= arguments.length ? __slice.call(arguments, 0, _i = arguments.length - 1) : (_i = 0, []), advice = arguments[_i++];
+ for (_j = 0, _len = functionNames.length; _j < _len; _j++) {
+ functionName = functionNames[_j];
+ object[functionName] = (__bind(function() {
+ var fn;
+ fn = object[functionName];
+ return function() {
+ var args, proceed;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ proceed = function() {
+ var proceededArgs;
+ proceededArgs = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ return fn.apply(object, proceededArgs);
+ };
+ return advice.apply(null, [proceed].concat(__slice.call(args)));
+ };
+ }, this))();
+ }
+ return combinator;
+ },
+ after: __bind(function(advice) {
+ var functionName, _i, _len;
+ for (_i = 0, _len = functionNames.length; _i < _len; _i++) {
+ functionName = functionNames[_i];
+ object[functionName] = (__bind(function() {
+ var fn;
+ fn = object[functionName];
+ return function() {
+ var args, result;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ result = fn.apply(object, args);
+ advice.apply(null, args);
+ return result;
+ };
+ }, this))();
+ }
+ return combinator;
+ }, this),
+ beforeAnyCallback: __bind(function(advice) {
+ var functionName, _i, _len;
+ for (_i = 0, _len = functionNames.length; _i < _len; _i++) {
+ functionName = functionNames[_i];
+ object[functionName] = (function() {
+ var fn;
+ fn = object[functionName];
+ return function() {
+ var adviced_args, args;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ adviced_args = [];
+ _.each(args, function(arg) {
+ var new_arg;
+ new_arg = _.isFunction(arg) ? function() {
+ var callback_args;
+ callback_args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ advice.apply(null, callback_args);
+ return arg.apply(null, callback_args);
+ } : arg;
+ return adviced_args.push(new_arg);
+ });
+ return fn.apply(object, adviced_args);
+ };
+ })();
+ }
+ return combinator;
+ }, this),
+ afterAnyCallback: __bind(function(advice) {
+ var functionName, _i, _len;
+ for (_i = 0, _len = functionNames.length; _i < _len; _i++) {
+ functionName = functionNames[_i];
+ object[functionName] = (function() {
+ var fn;
+ fn = object[functionName];
+ return function() {
+ var adviced_args, args;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ adviced_args = [];
+ _.each(args, function(arg) {
+ var new_arg;
+ new_arg = _.isFunction(arg) ? function() {
+ var callback_args, result;
+ callback_args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ result = arg.apply(null, callback_args);
+ advice.apply(null, callback_args);
+ return result;
+ } : void 0;
+ arg;
+ return adviced_args.push(new_arg);
+ });
+ return fn.apply(object, adviced_args);
+ };
+ })();
+ }
+ return combinator;
+ }, this)
+ };
+ }, this)
+ });
+}).call(this);
@@ -0,0 +1,105 @@
+(function() {
+ var __slice = Array.prototype.slice, __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+ _.defaults(this, {
+ YouAreDaChef: function() {
+ var clazzes, combinator, pointcuts;
+ clazzes = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ pointcuts = function(pointcut_exprs) {
+ return _.tap({}, function(cuts) {
+ return _.each(clazzes, function(clazz) {
+ return _.each(pointcut_exprs, function(expr) {
+ var name;
+ if (_.isString(expr) && _.isFunction(clazz.prototype[expr])) {
+ name = expr;
+ return cuts[name] = {
+ clazz: clazz,
+ pointcut: clazz.prototype[name],
+ inject: []
+ };
+ } else if (expr instanceof RegExp) {
+ return _.each(_.functions(clazz.prototype), function(name) {
+ var match;
+ if (match = name.match(expr)) {
+ return cuts[name] = {
+ clazz: clazz,
+ pointcut: clazz.prototype[name],
+ inject: match
+ };
+ }
+ });
+ }
+ });
+ });
+ });
+ };
+ return combinator = {
+ before: function() {
+ var advice, pointcut_exprs, _i;
+ pointcut_exprs = 2 <= arguments.length ? __slice.call(arguments, 0, _i = arguments.length - 1) : (_i = 0, []), advice = arguments[_i++];
+ _.each(pointcuts(pointcut_exprs), function(_arg, name) {
+ var clazz, inject, pointcut;
+ clazz = _arg.clazz, pointcut = _arg.pointcut, inject = _arg.inject;
+ return clazz.prototype[name] = function() {
+ var args;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ advice.call.apply(advice, [this].concat(__slice.call(inject), __slice.call(args)));
+ return pointcut.apply(this, args);
+ };
+ });
+ return combinator;
+ },
+ after: function() {
+ var advice, pointcut_exprs, _i;
+ pointcut_exprs = 2 <= arguments.length ? __slice.call(arguments, 0, _i = arguments.length - 1) : (_i = 0, []), advice = arguments[_i++];
+ _.each(pointcuts(pointcut_exprs), function(_arg, name) {
+ var clazz, inject, pointcut;
+ clazz = _arg.clazz, pointcut = _arg.pointcut, inject = _arg.inject;
+ return clazz.prototype[name] = function() {
+ var args;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ return _.tap(pointcut.apply(this, args), __bind(function() {
+ return advice.call.apply(advice, [this].concat(__slice.call(inject), __slice.call(args)));
+ }, this));
+ };
+ });
+ return combinator;
+ },
+ around: function() {
+ var advice, pointcut_exprs, _i;
+ pointcut_exprs = 2 <= arguments.length ? __slice.call(arguments, 0, _i = arguments.length - 1) : (_i = 0, []), advice = arguments[_i++];
+ _.each(pointcuts(pointcut_exprs), function(_arg, name) {
+ var clazz, inject, pointcut;
+ clazz = _arg.clazz, pointcut = _arg.pointcut, inject = _arg.inject;
+ return clazz.prototype[name] = function() {
+ var args, bound_pointcut;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ bound_pointcut = __bind(function() {
+ var args2;
+ args2 = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ return pointcut.apply(this, args2);
+ }, this);
+ return advice.call.apply(advice, [this, bound_pointcut].concat(__slice.call(inject), __slice.call(args)));
+ };
+ });
+ return combinator;
+ },
+ guard: function() {
+ var advice, pointcut_exprs, _i;
+ pointcut_exprs = 2 <= arguments.length ? __slice.call(arguments, 0, _i = arguments.length - 1) : (_i = 0, []), advice = arguments[_i++];
+ _.each(pointcuts(pointcut_exprs), function(_arg, name) {
+ var clazz, inject, pointcut;
+ clazz = _arg.clazz, pointcut = _arg.pointcut, inject = _arg.inject;
+ return clazz.prototype[name] = function() {
+ var args;
+ args = 1 <= arguments.length ? __slice.call(arguments, 0) : [];
+ if (advice.call.apply(advice, [this].concat(__slice.call(inject), __slice.call(args)))) {
+ return pointcut.apply(this, args);
+ }
+ };
+ });
+ return combinator;
+ }
+ };
+ }
+ });
+}).call(this);
@@ -0,0 +1,7 @@
+(function( window ) {
+ 'use strict';
+
+ if ( location.hostname === 'todomvc.com' ) {
+ var _gaq=[['_setAccount','UA-31081062-1'],['_trackPageview']];(function(d,t){var g=d.createElement(t),s=d.getElementsByTagName(t)[0];g.src='#';s.parentNode.insertBefore(g,s)}(document,'script'));
+ }
+})( window );
\ No newline at end of file
@@ -0,0 +1,1550 @@
+// lib/handlebars/base.js
+var Handlebars = {};
+
+Handlebars.VERSION = "1.0.beta.6";
+
+Handlebars.helpers = {};
+Handlebars.partials = {};
+
+Handlebars.registerHelper = function(name, fn, inverse) {
+ if(inverse) { fn.not = inverse; }
+ this.helpers[name] = fn;
+};
+
+Handlebars.registerPartial = function(name, str) {
+ this.partials[name] = str;
+};
+
+Handlebars.registerHelper('helperMissing', function(arg) {
+ if(arguments.length === 2) {
+ return undefined;
+ } else {
+ throw new Error("Could not find property '" + arg + "'");
+ }
+});
+
+var toString = Object.prototype.toString, functionType = "[object Function]";
+
+Handlebars.registerHelper('blockHelperMissing', function(context, options) {
+ var inverse = options.inverse || function() {}, fn = options.fn;
+
+
+ var ret = "";
+ var type = toString.call(context);
+
+ if(type === functionType) { context = context.call(this); }
+
+ if(context === true) {
+ return fn(this);
+ } else if(context === false || context == null) {
+ return inverse(this);
+ } else if(type === "[object Array]") {
+ if(context.length > 0) {
+ for(var i=0, j=context.length; i<j; i++) {
+ ret = ret + fn(context[i]);
+ }
+ } else {
+ ret = inverse(this);
+ }
+ return ret;
+ } else {
+ return fn(context);
+ }
+});
+
+Handlebars.registerHelper('each', function(context, options) {
+ var fn = options.fn, inverse = options.inverse;
+ var ret = "";
+
+ if(context && context.length > 0) {
+ for(var i=0, j=context.length; i<j; i++) {
+ ret = ret + fn(context[i]);
+ }
+ } else {
+ ret = inverse(this);
+ }
+ return ret;
+});
+
+Handlebars.registerHelper('if', function(context, options) {
+ var type = toString.call(context);
+ if(type === functionType) { context = context.call(this); }
+
+ if(!context || Handlebars.Utils.isEmpty(context)) {
+ return options.inverse(this);
+ } else {
+ return options.fn(this);
+ }
+});
+
+Handlebars.registerHelper('unless', function(context, options) {
+ var fn = options.fn, inverse = options.inverse;
+ options.fn = inverse;
+ options.inverse = fn;
+
+ return Handlebars.helpers['if'].call(this, context, options);
+});
+
+Handlebars.registerHelper('with', function(context, options) {
+ return options.fn(context);
+});
+
+Handlebars.registerHelper('log', function(context) {
+ Handlebars.log(context);
+});
+;
+// lib/handlebars/compiler/parser.js
+/* Jison generated parser */
+var handlebars = (function(){
+
+var parser = {trace: function trace() { },
+yy: {},
+symbols_: {"error":2,"root":3,"program":4,"EOF":5,"statements":6,"simpleInverse":7,"statement":8,"openInverse":9,"closeBlock":10,"openBlock":11,"mustache":12,"partial":13,"CONTENT":14,"COMMENT":15,"OPEN_BLOCK":16,"inMustache":17,"CLOSE":18,"OPEN_INVERSE":19,"OPEN_ENDBLOCK":20,"path":21,"OPEN":22,"OPEN_UNESCAPED":23,"OPEN_PARTIAL":24,"params":25,"hash":26,"param":27,"STRING":28,"INTEGER":29,"BOOLEAN":30,"hashSegments":31,"hashSegment":32,"ID":33,"EQUALS":34,"pathSegments":35,"SEP":36,"$accept":0,"$end":1},
+terminals_: {2:"error",5:"EOF",14:"CONTENT",15:"COMMENT",16:"OPEN_BLOCK",18:"CLOSE",19:"OPEN_INVERSE",20:"OPEN_ENDBLOCK",22:"OPEN",23:"OPEN_UNESCAPED",24:"OPEN_PARTIAL",28:"STRING",29:"INTEGER",30:"BOOLEAN",33:"ID",34:"EQUALS",36:"SEP"},
+productions_: [0,[3,2],[4,3],[4,1],[4,0],[6,1],[6,2],[8,3],[8,3],[8,1],[8,1],[8,1],[8,1],[11,3],[9,3],[10,3],[12,3],[12,3],[13,3],[13,4],[7,2],[17,3],[17,2],[17,2],[17,1],[25,2],[25,1],[27,1],[27,1],[27,1],[27,1],[26,1],[31,2],[31,1],[32,3],[32,3],[32,3],[32,3],[21,1],[35,3],[35,1]],
+performAction: function anonymous(yytext,yyleng,yylineno,yy,yystate,$$,_$) {
+
+var $0 = $$.length - 1;
+switch (yystate) {
+case 1: return $$[$0-1]
+break;
+case 2: this.$ = new yy.ProgramNode($$[$0-2], $$[$0])
+break;
+case 3: this.$ = new yy.ProgramNode($$[$0])
+break;
+case 4: this.$ = new yy.ProgramNode([])
+break;
+case 5: this.$ = [$$[$0]]
+break;
+case 6: $$[$0-1].push($$[$0]); this.$ = $$[$0-1]
+break;
+case 7: this.$ = new yy.InverseNode($$[$0-2], $$[$0-1], $$[$0])
+break;
+case 8: this.$ = new yy.BlockNode($$[$0-2], $$[$0-1], $$[$0])
+break;
+case 9: this.$ = $$[$0]
+break;
+case 10: this.$ = $$[$0]
+break;
+case 11: this.$ = new yy.ContentNode($$[$0])
+break;
+case 12: this.$ = new yy.CommentNode($$[$0])
+break;
+case 13: this.$ = new yy.MustacheNode($$[$0-1][0], $$[$0-1][1])
+break;
+case 14: this.$ = new yy.MustacheNode($$[$0-1][0], $$[$0-1][1])
+break;
+case 15: this.$ = $$[$0-1]
+break;
+case 16: this.$ = new yy.MustacheNode($$[$0-1][0], $$[$0-1][1])
+break;
+case 17: this.$ = new yy.MustacheNode($$[$0-1][0], $$[$0-1][1], true)
+break;
+case 18: this.$ = new yy.PartialNode($$[$0-1])
+break;
+case 19: this.$ = new yy.PartialNode($$[$0-2], $$[$0-1])
+break;
+case 20:
+break;
+case 21: this.$ = [[$$[$0-2]].concat($$[$0-1]), $$[$0]]
+break;
+case 22: this.$ = [[$$[$0-1]].concat($$[$0]), null]
+break;
+case 23: this.$ = [[$$[$0-1]], $$[$0]]
+break;
+case 24: this.$ = [[$$[$0]], null]
+break;
+case 25: $$[$0-1].push($$[$0]); this.$ = $$[$0-1];
+break;
+case 26: this.$ = [$$[$0]]
+break;
+case 27: this.$ = $$[$0]
+break;
+case 28: this.$ = new yy.StringNode($$[$0])
+break;
+case 29: this.$ = new yy.IntegerNode($$[$0])
+break;
+case 30: this.$ = new yy.BooleanNode($$[$0])
+break;
+case 31: this.$ = new yy.HashNode($$[$0])
+break;
+case 32: $$[$0-1].push($$[$0]); this.$ = $$[$0-1]
+break;
+case 33: this.$ = [$$[$0]]
+break;
+case 34: this.$ = [$$[$0-2], $$[$0]]
+break;
+case 35: this.$ = [$$[$0-2], new yy.StringNode($$[$0])]
+break;
+case 36: this.$ = [$$[$0-2], new yy.IntegerNode($$[$0])]
+break;
+case 37: this.$ = [$$[$0-2], new yy.BooleanNode($$[$0])]
+break;
+case 38: this.$ = new yy.IdNode($$[$0])
+break;
+case 39: $$[$0-2].push($$[$0]); this.$ = $$[$0-2];
+break;
+case 40: this.$ = [$$[$0]]
+break;
+}
+},
+table: [{3:1,4:2,5:[2,4],6:3,8:4,9:5,11:6,12:7,13:8,14:[1,9],15:[1,10],16:[1,12],19:[1,11],22:[1,13],23:[1,14],24:[1,15]},{1:[3]},{5:[1,16]},{5:[2,3],7:17,8:18,9:5,11:6,12:7,13:8,14:[1,9],15:[1,10],16:[1,12],19:[1,19],20:[2,3],22:[1,13],23:[1,14],24:[1,15]},{5:[2,5],14:[2,5],15:[2,5],16:[2,5],19:[2,5],20:[2,5],22:[2,5],23:[2,5],24:[2,5]},{4:20,6:3,8:4,9:5,11:6,12:7,13:8,14:[1,9],15:[1,10],16:[1,12],19:[1,11],20:[2,4],22:[1,13],23:[1,14],24:[1,15]},{4:21,6:3,8:4,9:5,11:6,12:7,13:8,14:[1,9],15:[1,10],16:[1,12],19:[1,11],20:[2,4],22:[1,13],23:[1,14],24:[1,15]},{5:[2,9],14:[2,9],15:[2,9],16:[2,9],19:[2,9],20:[2,9],22:[2,9],23:[2,9],24:[2,9]},{5:[2,10],14:[2,10],15:[2,10],16:[2,10],19:[2,10],20:[2,10],22:[2,10],23:[2,10],24:[2,10]},{5:[2,11],14:[2,11],15:[2,11],16:[2,11],19:[2,11],20:[2,11],22:[2,11],23:[2,11],24:[2,11]},{5:[2,12],14:[2,12],15:[2,12],16:[2,12],19:[2,12],20:[2,12],22:[2,12],23:[2,12],24:[2,12]},{17:22,21:23,33:[1,25],35:24},{17:26,21:23,33:[1,25],35:24},{17:27,21:23,33:[1,25],35:24},{17:28,21:23,33:[1,25],35:24},{21:29,33:[1,25],35:24},{1:[2,1]},{6:30,8:4,9:5,11:6,12:7,13:8,14:[1,9],15:[1,10],16:[1,12],19:[1,11],22:[1,13],23:[1,14],24:[1,15]},{5:[2,6],14:[2,6],15:[2,6],16:[2,6],19:[2,6],20:[2,6],22:[2,6],23:[2,6],24:[2,6]},{17:22,18:[1,31],21:23,33:[1,25],35:24},{10:32,20:[1,33]},{10:34,20:[1,33]},{18:[1,35]},{18:[2,24],21:40,25:36,26:37,27:38,28:[1,41],29:[1,42],30:[1,43],31:39,32:44,33:[1,45],35:24},{18:[2,38],28:[2,38],29:[2,38],30:[2,38],33:[2,38],36:[1,46]},{18:[2,40],28:[2,40],29:[2,40],30:[2,40],33:[2,40],36:[2,40]},{18:[1,47]},{18:[1,48]},{18:[1,49]},{18:[1,50],21:51,33:[1,25],35:24},{5:[2,2],8:18,9:5,11:6,12:7,13:8,14:[1,9],15:[1,10],16:[1,12],19:[1,11],20:[2,2],22:[1,13],23:[1,14],24:[1,15]},{14:[2,20],15:[2,20],16:[2,20],19:[2,20],22:[2,20],23:[2,20],24:[2,20]},{5:[2,7],14:[2,7],15:[2,7],16:[2,7],19:[2,7],20:[2,7],22:[2,7],23:[2,7],24:[2,7]},{21:52,33:[1,25],35:24},{5:[2,8],14:[2,8],15:[2,8],16:[2,8],19:[2,8],20:[2,8],22:[2,8],23:[2,8],24:[2,8]},{14:[2,14],15:[2,14],16:[2,14],19:[2,14],20:[2,14],22:[2,14],23:[2,14],24:[2,14]},{18:[2,22],21:40,26:53,27:54,28:[1,41],29:[1,42],30:[1,43],31:39,32:44,33:[1,45],35:24},{18:[2,23]},{18:[2,26],28:[2,26],29:[2,26],30:[2,26],33:[2,26]},{18:[2,31],32:55,33:[1,56]},{18:[2,27],28:[2,27],29:[2,27],30:[2,27],33:[2,27]},{18:[2,28],28:[2,28],29:[2,28],30:[2,28],33:[2,28]},{18:[2,29],28:[2,29],29:[2,29],30:[2,29],33:[2,29]},{18:[2,30],28:[2,30],29:[2,30],30:[2,30],33:[2,30]},{18:[2,33],33:[2,33]},{18:[2,40],28:[2,40],29:[2,40],30:[2,40],33:[2,40],34:[1,57],36:[2,40]},{33:[1,58]},{14:[2,13],15:[2,13],16:[2,13],19:[2,13],20:[2,13],22:[2,13],23:[2,13],24:[2,13]},{5:[2,16],14:[2,16],15:[2,16],16:[2,16],19:[2,16],20:[2,16],22:[2,16],23:[2,16],24:[2,16]},{5:[2,17],14:[2,17],15:[2,17],16:[2,17],19:[2,17],20:[2,17],22:[2,17],23:[2,17],24:[2,17]},{5:[2,18],14:[2,18],15:[2,18],16:[2,18],19:[2,18],20:[2,18],22:[2,18],23:[2,18],24:[2,18]},{18:[1,59]},{18:[1,60]},{18:[2,21]},{18:[2,25],28:[2,25],29:[2,25],30:[2,25],33:[2,25]},{18:[2,32],33:[2,32]},{34:[1,57]},{21:61,28:[1,62],29:[1,63],30:[1,64],33:[1,25],35:24},{18:[2,39],28:[2,39],29:[2,39],30:[2,39],33:[2,39],36:[2,39]},{5:[2,19],14:[2,19],15:[2,19],16:[2,19],19:[2,19],20:[2,19],22:[2,19],23:[2,19],24:[2,19]},{5:[2,15],14:[2,15],15:[2,15],16:[2,15],19:[2,15],20:[2,15],22:[2,15],23:[2,15],24:[2,15]},{18:[2,34],33:[2,34]},{18:[2,35],33:[2,35]},{18:[2,36],33:[2,36]},{18:[2,37],33:[2,37]}],
+defaultActions: {16:[2,1],37:[2,23],53:[2,21]},
+parseError: function parseError(str, hash) {
+ throw new Error(str);
+},
+parse: function parse(input) {
+ var self = this, stack = [0], vstack = [null], lstack = [], table = this.table, yytext = "", yylineno = 0, yyleng = 0, recovering = 0, TERROR = 2, EOF = 1;
+ this.lexer.setInput(input);
+ this.lexer.yy = this.yy;
+ this.yy.lexer = this.lexer;
+ if (typeof this.lexer.yylloc == "undefined")
+ this.lexer.yylloc = {};
+ var yyloc = this.lexer.yylloc;
+ lstack.push(yyloc);
+ if (typeof this.yy.parseError === "function")
+ this.parseError = this.yy.parseError;
+ function popStack(n) {
+ stack.length = stack.length - 2 * n;
+ vstack.length = vstack.length - n;
+ lstack.length = lstack.length - n;
+ }
+ function lex() {
+ var token;
+ token = self.lexer.lex() || 1;
+ if (typeof token !== "number") {
+ token = self.symbols_[token] || token;
+ }
+ return token;
+ }
+ var symbol, preErrorSymbol, state, action, a, r, yyval = {}, p, len, newState, expected;
+ while (true) {
+ state = stack[stack.length - 1];
+ if (this.defaultActions[state]) {
+ action = this.defaultActions[state];
+ } else {
+ if (symbol == null)
+ symbol = lex();
+ action = table[state] && table[state][symbol];
+ }
+ if (typeof action === "undefined" || !action.length || !action[0]) {
+ if (!recovering) {
+ expected = [];
+ for (p in table[state])
+ if (this.terminals_[p] && p > 2) {
+ expected.push("'" + this.terminals_[p] + "'");
+ }
+ var errStr = "";
+ if (this.lexer.showPosition) {
+ errStr = "Parse error on line " + (yylineno + 1) + ":\n" + this.lexer.showPosition() + "\nExpecting " + expected.join(", ") + ", got '" + this.terminals_[symbol] + "'";
+ } else {
+ errStr = "Parse error on line " + (yylineno + 1) + ": Unexpected " + (symbol == 1?"end of input":"'" + (this.terminals_[symbol] || symbol) + "'");
+ }
+ this.parseError(errStr, {text: this.lexer.match, token: this.terminals_[symbol] || symbol, line: this.lexer.yylineno, loc: yyloc, expected: expected});
+ }
+ }
+ if (action[0] instanceof Array && action.length > 1) {
+ throw new Error("Parse Error: multiple actions possible at state: " + state + ", token: " + symbol);
+ }
+ switch (action[0]) {
+ case 1:
+ stack.push(symbol);
+ vstack.push(this.lexer.yytext);
+ lstack.push(this.lexer.yylloc);
+ stack.push(action[1]);
+ symbol = null;
+ if (!preErrorSymbol) {
+ yyleng = this.lexer.yyleng;
+ yytext = this.lexer.yytext;
+ yylineno = this.lexer.yylineno;
+ yyloc = this.lexer.yylloc;
+ if (recovering > 0)
+ recovering--;
+ } else {
+ symbol = preErrorSymbol;
+ preErrorSymbol = null;
+ }
+ break;
+ case 2:
+ len = this.productions_[action[1]][1];
+ yyval.$ = vstack[vstack.length - len];
+ yyval._$ = {first_line: lstack[lstack.length - (len || 1)].first_line, last_line: lstack[lstack.length - 1].last_line, first_column: lstack[lstack.length - (len || 1)].first_column, last_column: lstack[lstack.length - 1].last_column};
+ r = this.performAction.call(yyval, yytext, yyleng, yylineno, this.yy, action[1], vstack, lstack);
+ if (typeof r !== "undefined") {
+ return r;
+ }
+ if (len) {
+ stack = stack.slice(0, -1 * len * 2);
+ vstack = vstack.slice(0, -1 * len);
+ lstack = lstack.slice(0, -1 * len);
+ }
+ stack.push(this.productions_[action[1]][0]);
+ vstack.push(yyval.$);
+ lstack.push(yyval._$);
+ newState = table[stack[stack.length - 2]][stack[stack.length - 1]];
+ stack.push(newState);
+ break;
+ case 3:
+ return true;
+ }
+ }
+ return true;
+}
+};/* Jison generated lexer */
+var lexer = (function(){
+
+var lexer = ({EOF:1,
+parseError:function parseError(str, hash) {
+ if (this.yy.parseError) {
+ this.yy.parseError(str, hash);
+ } else {
+ throw new Error(str);
+ }
+ },
+setInput:function (input) {
+ this._input = input;
+ this._more = this._less = this.done = false;
+ this.yylineno = this.yyleng = 0;
+ this.yytext = this.matched = this.match = '';
+ this.conditionStack = ['INITIAL'];
+ this.yylloc = {first_line:1,first_column:0,last_line:1,last_column:0};
+ return this;
+ },
+input:function () {
+ var ch = this._input[0];
+ this.yytext+=ch;
+ this.yyleng++;
+ this.match+=ch;
+ this.matched+=ch;
+ var lines = ch.match(/\n/);
+ if (lines) this.yylineno++;
+ this._input = this._input.slice(1);
+ return ch;
+ },
+unput:function (ch) {
+ this._input = ch + this._input;
+ return this;
+ },
+more:function () {
+ this._more = true;
+ return this;
+ },
+pastInput:function () {
+ var past = this.matched.substr(0, this.matched.length - this.match.length);
+ return (past.length > 20 ? '...':'') + past.substr(-20).replace(/\n/g, "");
+ },
+upcomingInput:function () {
+ var next = this.match;
+ if (next.length < 20) {
+ next += this._input.substr(0, 20-next.length);
+ }
+ return (next.substr(0,20)+(next.length > 20 ? '...':'')).replace(/\n/g, "");
+ },
+showPosition:function () {
+ var pre = this.pastInput();
+ var c = new Array(pre.length + 1).join("-");
+ return pre + this.upcomingInput() + "\n" + c+"^";
+ },
+next:function () {
+ if (this.done) {
+ return this.EOF;
+ }
+ if (!this._input) this.done = true;
+
+ var token,
+ match,
+ col,
+ lines;
+ if (!this._more) {
+ this.yytext = '';
+ this.match = '';
+ }
+ var rules = this._currentRules();
+ for (var i=0;i < rules.length; i++) {
+ match = this._input.match(this.rules[rules[i]]);
+ if (match) {
+ lines = match[0].match(/\n.*/g);
+ if (lines) this.yylineno += lines.length;
+ this.yylloc = {first_line: this.yylloc.last_line,
+ last_line: this.yylineno+1,
+ first_column: this.yylloc.last_column,
+ last_column: lines ? lines[lines.length-1].length-1 : this.yylloc.last_column + match[0].length}
+ this.yytext += match[0];
+ this.match += match[0];
+ this.matches = match;
+ this.yyleng = this.yytext.length;
+ this._more = false;
+ this._input = this._input.slice(match[0].length);
+ this.matched += match[0];
+ token = this.performAction.call(this, this.yy, this, rules[i],this.conditionStack[this.conditionStack.length-1]);
+ if (token) return token;
+ else return;
+ }
+ }
+ if (this._input === "") {
+ return this.EOF;
+ } else {
+ this.parseError('Lexical error on line '+(this.yylineno+1)+'. Unrecognized text.\n'+this.showPosition(),
+ {text: "", token: null, line: this.yylineno});
+ }
+ },
+lex:function lex() {
+ var r = this.next();
+ if (typeof r !== 'undefined') {
+ return r;
+ } else {
+ return this.lex();
+ }
+ },
+begin:function begin(condition) {
+ this.conditionStack.push(condition);
+ },
+popState:function popState() {
+ return this.conditionStack.pop();
+ },
+_currentRules:function _currentRules() {
+ return this.conditions[this.conditionStack[this.conditionStack.length-1]].rules;
+ },
+topState:function () {
+ return this.conditionStack[this.conditionStack.length-2];
+ },
+pushState:function begin(condition) {
+ this.begin(condition);
+ }});
+lexer.performAction = function anonymous(yy,yy_,$avoiding_name_collisions,YY_START) {
+
+var YYSTATE=YY_START
+switch($avoiding_name_collisions) {
+case 0:
+ if(yy_.yytext.slice(-1) !== "\\") this.begin("mu");
+ if(yy_.yytext.slice(-1) === "\\") yy_.yytext = yy_.yytext.substr(0,yy_.yyleng-1), this.begin("emu");
+ if(yy_.yytext) return 14;
+
+break;
+case 1: return 14;
+break;
+case 2: this.popState(); return 14;
+break;
+case 3: return 24;
+break;
+case 4: return 16;
+break;
+case 5: return 20;
+break;
+case 6: return 19;
+break;
+case 7: return 19;
+break;
+case 8: return 23;
+break;
+case 9: return 23;
+break;
+case 10: yy_.yytext = yy_.yytext.substr(3,yy_.yyleng-5); this.popState(); return 15;
+break;
+case 11: return 22;
+break;
+case 12: return 34;
+break;
+case 13: return 33;
+break;
+case 14: return 33;
+break;
+case 15: return 36;
+break;
+case 16: /*ignore whitespace*/
+break;
+case 17: this.popState(); return 18;
+break;
+case 18: this.popState(); return 18;
+break;
+case 19: yy_.yytext = yy_.yytext.substr(1,yy_.yyleng-2).replace(/\\"/g,'"'); return 28;
+break;
+case 20: return 30;
+break;
+case 21: return 30;
+break;
+case 22: return 29;
+break;
+case 23: return 33;
+break;
+case 24: yy_.yytext = yy_.yytext.substr(1, yy_.yyleng-2); return 33;
+break;
+case 25: return 'INVALID';
+break;
+case 26: return 5;
+break;
+}
+};
+lexer.rules = [/^[^\x00]*?(?=(\{\{))/,/^[^\x00]+/,/^[^\x00]{2,}?(?=(\{\{))/,/^\{\{>/,/^\{\{#/,/^\{\{\//,/^\{\{\^/,/^\{\{\s*else\b/,/^\{\{\{/,/^\{\{&/,/^\{\{![\s\S]*?\}\}/,/^\{\{/,/^=/,/^\.(?=[} ])/,/^\.\./,/^[\/.]/,/^\s+/,/^\}\}\}/,/^\}\}/,/^"(\\["]|[^"])*"/,/^true(?=[}\s])/,/^false(?=[}\s])/,/^[0-9]+(?=[}\s])/,/^[a-zA-Z0-9_$-]+(?=[=}\s\/.])/,/^\[[^\]]*\]/,/^./,/^$/];
+lexer.conditions = {"mu":{"rules":[3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26],"inclusive":false},"emu":{"rules":[2],"inclusive":false},"INITIAL":{"rules":[0,1,26],"inclusive":true}};return lexer;})()
+parser.lexer = lexer;
+return parser;
+})();
+if (typeof require !== 'undefined' && typeof exports !== 'undefined') {
+exports.parser = handlebars;
+exports.parse = function () { return handlebars.parse.apply(handlebars, arguments); }
+exports.main = function commonjsMain(args) {
+ if (!args[1])
+ throw new Error('Usage: '+args[0]+' FILE');
+ if (typeof process !== 'undefined') {
+ var source = require('fs').readFileSync(require('path').join(process.cwd(), args[1]), "utf8");
+ } else {
+ var cwd = require("file").path(require("file").cwd());
+ var source = cwd.join(args[1]).read({charset: "utf-8"});
+ }
+ return exports.parser.parse(source);
+}
+if (typeof module !== 'undefined' && require.main === module) {
+ exports.main(typeof process !== 'undefined' ? process.argv.slice(1) : require("system").args);
+}
+};
+;
+// lib/handlebars/compiler/base.js
+Handlebars.Parser = handlebars;
+
+Handlebars.parse = function(string) {
+ Handlebars.Parser.yy = Handlebars.AST;
+ return Handlebars.Parser.parse(string);
+};
+
+Handlebars.print = function(ast) {
+ return new Handlebars.PrintVisitor().accept(ast);
+};
+
+Handlebars.logger = {
+ DEBUG: 0, INFO: 1, WARN: 2, ERROR: 3, level: 3,
+
+ // override in the host environment
+ log: function(level, str) {}
+};
+
+Handlebars.log = function(level, str) { Handlebars.logger.log(level, str); };
+;
+// lib/handlebars/compiler/ast.js
+(function() {
+
+ Handlebars.AST = {};
+
+ Handlebars.AST.ProgramNode = function(statements, inverse) {
+ this.type = "program";
+ this.statements = statements;
+ if(inverse) { this.inverse = new Handlebars.AST.ProgramNode(inverse); }
+ };
+
+ Handlebars.AST.MustacheNode = function(params, hash, unescaped) {
+ this.type = "mustache";
+ this.id = params[0];
+ this.params = params.slice(1);
+ this.hash = hash;
+ this.escaped = !unescaped;
+ };
+
+ Handlebars.AST.PartialNode = function(id, context) {
+ this.type = "partial";
+
+ // TODO: disallow complex IDs
+
+ this.id = id;
+ this.context = context;
+ };
+
+ var verifyMatch = function(open, close) {
+ if(open.original !== close.original) {
+ throw new Handlebars.Exception(open.original + " doesn't match " + close.original);
+ }
+ };
+
+ Handlebars.AST.BlockNode = function(mustache, program, close) {
+ verifyMatch(mustache.id, close);
+ this.type = "block";
+ this.mustache = mustache;
+ this.program = program;