Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with
or
.
Download ZIP
Browse files

initial commit.

  • Loading branch information...
commit 38fb3d0b34cc423a2140b9863f7c9acfcdda1c0a 1 parent 8a362eb
@addyosmani authored
Showing with 1,551 additions and 2 deletions.
  1. +23 −0 LICENSE
  2. +2 −2 README.md
  3. +62 −0 Rakefile
  4. +75 −0 TESTING.md
  5. +436 −0 mustache.js
  6. +6 −0 mustache.min.js
  7. +3 −0  spec/_files/ampersand_escape.js
  8. +1 −0  spec/_files/ampersand_escape.mustache
  9. +1 −0  spec/_files/ampersand_escape.txt
  10. +1 −0  spec/_files/apostrophe.js
  11. +1 −0  spec/_files/apostrophe.mustache
  12. +1 −0  spec/_files/apostrophe.txt
  13. +1 −0  spec/_files/array_of_partials_implicit_partial.2.mustache
  14. +3 −0  spec/_files/array_of_partials_implicit_partial.js
  15. +4 −0 spec/_files/array_of_partials_implicit_partial.mustache
  16. +5 −0 spec/_files/array_of_partials_implicit_partial.txt
  17. +1 −0  spec/_files/array_of_partials_partial.2.mustache
  18. +3 −0  spec/_files/array_of_partials_partial.js
  19. +4 −0 spec/_files/array_of_partials_partial.mustache
  20. +5 −0 spec/_files/array_of_partials_partial.txt
  21. +1 −0  spec/_files/array_of_strings.js
  22. +1 −0  spec/_files/array_of_strings.mustache
  23. +1 −0  spec/_files/array_of_strings.txt
  24. +1 −0  spec/_files/array_of_strings_options.js
  25. +2 −0  spec/_files/array_of_strings_options.mustache
  26. +1 −0  spec/_files/array_of_strings_options.txt
  27. +4 −0 spec/_files/array_partial.2.mustache
  28. +5 −0 spec/_files/array_partial.js
  29. +1 −0  spec/_files/array_partial.mustache
  30. +6 −0 spec/_files/array_partial.txt
  31. +3 −0  spec/_files/bug_11_eating_whitespace.js
  32. +1 −0  spec/_files/bug_11_eating_whitespace.mustache
  33. +1 −0  spec/_files/bug_11_eating_whitespace.txt
  34. +5 −0 spec/_files/comments.js
  35. +1 −0  spec/_files/comments.mustache
  36. +1 −0  spec/_files/comments.txt
  37. +19 −0 spec/_files/complex.js
  38. +16 −0 spec/_files/complex.mustache
  39. +6 −0 spec/_files/complex.txt
  40. +6 −0 spec/_files/delimiters.js
  41. +7 −0 spec/_files/delimiters.mustache
  42. +5 −0 spec/_files/delimiters.txt
  43. +23 −0 spec/_files/dot_notation.js
  44. +9 −0 spec/_files/dot_notation.mustache
  45. +9 −0 spec/_files/dot_notation.txt
  46. +5 −0 spec/_files/double_render.js
  47. +1 −0  spec/_files/double_render.mustache
  48. +1 −0  spec/_files/double_render.txt
  49. +1 −0  spec/_files/empty_partial.2.mustache
  50. +3 −0  spec/_files/empty_partial.js
  51. +2 −0  spec/_files/empty_partial.mustache
  52. +2 −0  spec/_files/empty_partial.txt
  53. +1 −0  spec/_files/empty_sections.js
  54. +1 −0  spec/_files/empty_sections.mustache
  55. +1 −0  spec/_files/empty_sections.txt
  56. +1 −0  spec/_files/empty_template.js
  57. +1 −0  spec/_files/empty_template.mustache
  58. +1 −0  spec/_files/empty_template.txt
  59. +1 −0  spec/_files/error_not_found.js
  60. +1 −0  spec/_files/error_not_found.mustache
  61. +1 −0  spec/_files/error_not_found.txt
  62. +7 −0 spec/_files/escaped.js
  63. +3 −0  spec/_files/escaped.mustache
  64. +3 −0  spec/_files/escaped.txt
  65. +9 −0 spec/_files/higher_order_sections.js
  66. +1 −0  spec/_files/higher_order_sections.mustache
  67. +1 −0  spec/_files/higher_order_sections.txt
  68. +3 −0  spec/_files/inverted_section.js
  69. +2 −0  spec/_files/inverted_section.mustache
  70. +1 −0  spec/_files/inverted_section.txt
  71. +5 −0 spec/_files/keys_with_questionmarks.js
  72. +3 −0  spec/_files/keys_with_questionmarks.mustache
  73. +1 −0  spec/_files/keys_with_questionmarks.txt
  74. +7 −0 spec/_files/nesting.js
  75. +5 −0 spec/_files/nesting.mustache
  76. +4 −0 spec/_files/nesting.txt
  77. +9 −0 spec/_files/null_string.js
  78. +5 −0 spec/_files/null_string.mustache
  79. +5 −0 spec/_files/null_string.txt
  80. +4 −0 spec/_files/partial_recursion.2.mustache
  81. +11 −0 spec/_files/partial_recursion.js
  82. +4 −0 spec/_files/partial_recursion.mustache
  83. +3 −0  spec/_files/partial_recursion.txt
  84. +8 −0 spec/_files/recursion_with_same_names.js
  85. +7 −0 spec/_files/recursion_with_same_names.mustache
  86. +6 −0 spec/_files/recursion_with_same_names.txt
  87. +6 −0 spec/_files/reuse_of_enumerables.js
  88. +8 −0 spec/_files/reuse_of_enumerables.mustache
  89. +8 −0 spec/_files/reuse_of_enumerables.txt
  90. +7 −0 spec/_files/section_as_context.js
  91. +9 −0 spec/_files/section_as_context.mustache
  92. +6 −0 spec/_files/section_as_context.txt
  93. +8 −0 spec/_files/simple.js
  94. +5 −0 spec/_files/simple.mustache
  95. +3 −0  spec/_files/simple.txt
  96. +1 −0  spec/_files/template_partial.2.mustache
  97. +8 −0 spec/_files/template_partial.js
  98. +2 −0  spec/_files/template_partial.mustache
  99. +2 −0  spec/_files/template_partial.txt
  100. +4 −0 spec/_files/two_in_a_row.js
  101. +1 −0  spec/_files/two_in_a_row.mustache
  102. +1 −0  spec/_files/two_in_a_row.txt
  103. +1 −0  spec/_files/two_sections.js
  104. +4 −0 spec/_files/two_sections.mustache
  105. +1 −0  spec/_files/two_sections.txt
  106. +5 −0 spec/_files/unescaped.js
  107. +1 −0  spec/_files/unescaped.mustache
  108. +1 −0  spec/_files/unescaped.txt
  109. +1 −0  spec/_files/unknown_pragma.js
  110. +1 −0  spec/_files/unknown_pragma.mustache
  111. +1 −0  spec/_files/unknown_pragma.txt
  112. +5 −0 spec/_files/view_partial.2.mustache
  113. +19 −0 spec/_files/view_partial.js
  114. +3 −0  spec/_files/view_partial.mustache
  115. +6 −0 spec/_files/view_partial.txt
  116. +5 −0 spec/_files/whitespace_partial.2.mustache
  117. +19 −0 spec/_files/whitespace_partial.js
  118. +3 −0  spec/_files/whitespace_partial.mustache
  119. +6 −0 spec/_files/whitespace_partial.txt
  120. +276 −0 spec/mustache_spec.rb
  121. +8 −0 wrappers/commonjs/mustache.js.tpl.post
  122. +6 −0 wrappers/commonjs/mustache.js.tpl.pre
  123. +8 −0 wrappers/commonjs/package.json
  124. +4 −0 wrappers/dojo/mustache.js.tpl.post
  125. +9 −0 wrappers/dojo/mustache.js.tpl.pre
  126. +6 −0 wrappers/jquery/jquery.mustache.js.tpl.post
  127. +9 −0 wrappers/jquery/jquery.mustache.js.tpl.pre
  128. +9 −0 wrappers/qooxdoo/qooxdoo.mustache.js.tpl.post
  129. +127 −0 wrappers/qooxdoo/qooxdoo.mustache.js.tpl.pre
  130. +3 −0  wrappers/requirejs/requirejs.mustache.js.tpl.post
  131. +6 −0 wrappers/requirejs/requirejs.mustache.js.tpl.pre
  132. +4 −0 wrappers/yui3/mustache.js.tpl.post
  133. +1 −0  wrappers/yui3/mustache.js.tpl.pre
View
23 LICENSE
@@ -0,0 +1,23 @@
+The MIT License
+
+Copyright (c) 2009 Chris Wanstrath (Ruby)
+Copyright (c) 2010 Jan Lehnardt (JavaScript)
+
+Permission is hereby granted, free of charge, to any person obtaining
+a copy of this software and associated documentation files (the
+"Software"), to deal in the Software without restriction, including
+without limitation the rights to use, copy, modify, merge, publish,
+distribute, sublicense, and/or sell copies of the Software, and to
+permit persons to whom the Software is furnished to do so, subject to
+the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
View
4 README.md
@@ -2,10 +2,10 @@
## What is this?
-A special build of mustache.js (by @janl) which works within the security policies defined by
+A special build of [mustache.js](https://github.com/janl/mustache.js) (by [@janl](https://github.com/janl)) which works within the security policies defined by
the Google Chrome Apps/Extensions [Content Security Policy](http://code.google.com/chrome/extensions/contentSecurityPolicy.html).
-In CSP, inline JavaScript as well as potentially harmful string-to-JS methods such as `eval` are not executed. Whilst great for security, this also means that it can be a challenge finding a popular templating engine which doesn't use features disabled by the CSP.
+In CSP, inline JavaScript as well as potentially harmful string-to-JS methods such as `eval` are not executed. Whilst great for security, this also means that it can be a challenge finding a popular templating engine which doesn't use features disabled by the policy.
This version of mustache works fine under the CSP and has been used with frameworks such as Backbone.js within Chrome Apps without any issues.
View
62 Rakefile
@@ -0,0 +1,62 @@
+require 'rake'
+require 'rake/clean'
+
+task :default => :spec
+
+desc "Run all specs"
+task :spec do
+ require 'rspec/core/rake_task'
+ RSpec::Core::RakeTask.new(:spec) do |t|
+ #t.spec_opts = ['--options', "\"#{File.dirname(__FILE__)}/spec/spec.opts\""]
+ t.pattern = 'spec/*_spec.rb'
+ end
+end
+
+def version
+ File.read("mustache.js").match('version: "([^\"]+)",$')[1]
+end
+
+# Creates a rule that uses the .tmpl.{pre,post} stuff to make a final,
+# wrapped, output file. There is some extra complexity because Dojo and YUI3
+# use different template files and final locations.
+def templated_build(name, opts={})
+ short = name.downcase
+ source = File.join("wrappers", short)
+ dependencies = ["mustache.js"] + Dir.glob("#{source}/*.tpl.*")
+ target_js = opts[:location] ? "mustache.js" : "#{short}.mustache.js"
+
+ CLEAN.include(opts[:location] ? opts[:location] : target_js)
+
+ desc "Package for #{name}"
+ task short.to_sym => dependencies do
+ puts "Packaging for #{name}"
+
+ mkdir_p opts[:location] if opts[:location]
+
+ sh "cat #{source}/#{target_js}.tpl.pre mustache.js \
+ #{source}/#{target_js}.tpl.post > #{opts[:location] || '.'}/#{target_js}"
+
+ # extra
+ if opts[:extra]
+ sh "sed -e 's/{{version}}/#{version}/' #{source}/#{opts[:extra]} \
+ > #{opts[:location]}/#{opts[:extra]}"
+ end
+
+ puts "Done, see #{opts[:location] || '.'}/#{target_js}"
+ end
+end
+
+templated_build "CommonJS", :location => "lib", :extra => "package.json"
+templated_build "jQuery"
+templated_build "Dojo", :location => "dojox/string"
+templated_build "YUI3", :location => "yui3/mustache"
+templated_build "RequireJS"
+templated_build "qooxdoo"
+
+task :minify do
+ # npm install uglify-js
+ mmjs = "mustache.min.js"
+ `echo "/*! Version: 0.4.2 */" > #{mmjs}`
+ `uglifyjs mustache.js >> #{mmjs}`
+ puts "Created #{mmjs}"
+end
View
75 TESTING.md
@@ -0,0 +1,75 @@
+## Running the mustache.js Test Suite
+
+Notice: the tests are only expected to run on unixoid systems.
+
+The mustache.js test suite uses the [RSpec](http://rspec.info/) testing
+framework. In order to run the tests you'll need to install [Ruby](http://ruby-lang.org/)
+as well as the `rake`, `rspec` (>=2), and `json` [RubyGems](http://rubygems.org/).
+
+### How to install Ruby and the required gems from source
+
+Make sure you have the required tools to compile it:
+
+ $ apt-get install build-essential libssl-dev libreadline5-dev zlib1g-dev
+
+Download and extract the Ruby source, and install it:
+
+ $ wget ftp://ftp.ruby-lang.org/pub/ruby/stable-snapshot.tar.gz
+ $ tar xvzf stable-snapshot.tar.gz
+ $ cd ruby
+ $ ./configure && make && make install
+
+Download and extract RubyGems, and install it:
+
+ $ wget http://production.cf.rubygems.org/rubygems/rubygems-1.8.12.tgz
+ $ tar xzvf rubygems-1.8.12.tgz
+ $ cd rubygems-1.8.12
+ $ ruby setup.rb
+
+If you want to update RubyGems:
+
+ $ gem update --system
+
+Install the required gems:
+
+ $ gem install rake rspec json
+
+That's it!
+
+### How to install node.js from source
+
+ $ git clone https://github.com/joyent/node.git
+ $ cd node
+ $ # select the version to install, master is unstable;
+ $ # latest stable version is advertised on http://nodejs.org
+ $ git checkout v0.6.11
+ $ ./configure
+ $ make
+ $ sudo make install
+
+### How to run the tests
+
+The mustache.js test suite currently uses 4 different JavaScript runtime engines
+to maximize portability across platforms and browsers. They are:
+
+ * node
+ * SpiderMonkey (Mozilla, Firefox)
+ * JavaScriptCore (WebKit, Safari)
+ * Rhino (Mozilla, Java)
+
+When the test suite runs it will automatically determine which platforms are
+available on your machine and run on all of them. The suite must run on at least
+one platform in order to succeed.
+
+Once you have at least one JavaScript platform installed, you can run the test
+suite with the following command:
+
+ $ rake
+
+### How to create a test
+
+All test files live in the spec/_files directory. To create a new test:
+
+ * Create a template file `somename.mustache`
+ * Create a javascript file with data and functions `somename.js`
+ * Create a file the expected result `somename.txt`
View
436 mustache.js
@@ -0,0 +1,436 @@
+/*
+ mustache.js — Logic-less templates in JavaScript
+
+ See http://mustache.github.com/ for more info.
+*/
+
+var Mustache = function () {
+ var _toString = Object.prototype.toString;
+
+ Array.isArray = Array.isArray || function (obj) {
+ return _toString.call(obj) == "[object Array]";
+ }
+
+ var _trim = String.prototype.trim, trim;
+
+ if (_trim) {
+ trim = function (text) {
+ return text == null ? "" : _trim.call(text);
+ }
+ } else {
+ var trimLeft, trimRight;
+
+ // IE doesn't match non-breaking spaces with \s.
+ if ((/\S/).test("\xA0")) {
+ trimLeft = /^[\s\xA0]+/;
+ trimRight = /[\s\xA0]+$/;
+ } else {
+ trimLeft = /^\s+/;
+ trimRight = /\s+$/;
+ }
+
+ trim = function (text) {
+ return text == null ? "" :
+ text.toString().replace(trimLeft, "").replace(trimRight, "");
+ }
+ }
+
+ var escapeMap = {
+ "&": "&",
+ "<": "&lt;",
+ ">": "&gt;",
+ '"': '&quot;',
+ "'": '&#39;'
+ };
+
+ function escapeHTML(string) {
+ return String(string).replace(/&(?!#?\w+;)|[<>"']/g, function (s) {
+ return escapeMap[s] || s;
+ });
+ }
+
+ var regexCache = {};
+ var Renderer = function () {};
+
+ Renderer.prototype = {
+ otag: "{{",
+ ctag: "}}",
+ pragmas: {},
+ buffer: [],
+ pragmas_implemented: {
+ "IMPLICIT-ITERATOR": true
+ },
+ context: {},
+
+ render: function (template, context, partials, in_recursion) {
+ // reset buffer & set context
+ if (!in_recursion) {
+ this.context = context;
+ this.buffer = []; // TODO: make this non-lazy
+ }
+
+ // fail fast
+ if (!this.includes("", template)) {
+ if (in_recursion) {
+ return template;
+ } else {
+ this.send(template);
+ return;
+ }
+ }
+
+ // get the pragmas together
+ template = this.render_pragmas(template);
+
+ // render the template
+ var html = this.render_section(template, context, partials);
+
+ // render_section did not find any sections, we still need to render the tags
+ if (html === false) {
+ html = this.render_tags(template, context, partials, in_recursion);
+ }
+
+ if (in_recursion) {
+ return html;
+ } else {
+ this.sendLines(html);
+ }
+ },
+
+ /*
+ Sends parsed lines
+ */
+ send: function (line) {
+ if (line !== "") {
+ this.buffer.push(line);
+ }
+ },
+
+ sendLines: function (text) {
+ if (text) {
+ var lines = text.split("\n");
+ for (var i = 0; i < lines.length; i++) {
+ this.send(lines[i]);
+ }
+ }
+ },
+
+ /*
+ Looks for %PRAGMAS
+ */
+ render_pragmas: function (template) {
+ // no pragmas
+ if (!this.includes("%", template)) {
+ return template;
+ }
+
+ var that = this;
+ var regex = this.getCachedRegex("render_pragmas", function (otag, ctag) {
+ return new RegExp(otag + "%([\\w-]+) ?([\\w]+=[\\w]+)?" + ctag, "g");
+ });
+
+ return template.replace(regex, function (match, pragma, options) {
+ if (!that.pragmas_implemented[pragma]) {
+ throw({message:
+ "This implementation of mustache doesn't understand the '" +
+ pragma + "' pragma"});
+ }
+ that.pragmas[pragma] = {};
+ if (options) {
+ var opts = options.split("=");
+ that.pragmas[pragma][opts[0]] = opts[1];
+ }
+ return "";
+ // ignore unknown pragmas silently
+ });
+ },
+
+ /*
+ Tries to find a partial in the curent scope and render it
+ */
+ render_partial: function (name, context, partials) {
+ name = trim(name);
+ if (!partials || partials[name] === undefined) {
+ throw({message: "unknown_partial '" + name + "'"});
+ }
+ if (!context || typeof context[name] != "object") {
+ return this.render(partials[name], context, partials, true);
+ }
+ return this.render(partials[name], context[name], partials, true);
+ },
+
+ /*
+ Renders inverted (^) and normal (#) sections
+ */
+ render_section: function (template, context, partials) {
+ if (!this.includes("#", template) && !this.includes("^", template)) {
+ // did not render anything, there were no sections
+ return false;
+ }
+
+ var that = this;
+
+ var regex = this.getCachedRegex("render_section", function (otag, ctag) {
+ // This regex matches _the first_ section ({{#foo}}{{/foo}}), and captures the remainder
+ return new RegExp(
+ "^([\\s\\S]*?)" + // all the crap at the beginning that is not {{*}} ($1)
+
+ otag + // {{
+ "(\\^|\\#)\\s*(.+?)\\s*" +// #foo (# == $2, foo == $3), not greedy
+ ctag + // }}
+
+ "\n*([\\s\\S]*?)" + // between the tag ($2). leading newlines are dropped
+
+ otag + // {{
+ "\\/\\s*\\3\\s*" + // /foo (backreference to the opening tag).
+ ctag + // }}
+
+ "\\s*([\\s\\S]*)$", // everything else in the string ($4). leading whitespace is dropped.
+
+ "g");
+ });
+
+
+ // for each {{#foo}}{{/foo}} section do...
+ return template.replace(regex, function (match, before, type, name, content, after) {
+ // before contains only tags, no sections
+ var renderedBefore = before ? that.render_tags(before, context, partials, true) : "",
+
+ // after may contain both sections and tags, so use full rendering function
+ renderedAfter = after ? that.render(after, context, partials, true) : "",
+
+ // will be computed below
+ renderedContent,
+
+ value = that.find(name, context);
+
+ if (type === "^") { // inverted section
+ if (!value || Array.isArray(value) && value.length === 0) {
+ // false or empty list, render it
+ renderedContent = that.render(content, context, partials, true);
+ } else {
+ renderedContent = "";
+ }
+ } else if (type === "#") { // normal section
+ if (Array.isArray(value)) { // Enumerable, Let's loop!
+ renderedContent = that.map(value, function (row) {
+ return that.render(content, that.create_context(row), partials, true);
+ }).join("");
+ } else if (that.is_object(value)) { // Object, Use it as subcontext!
+ renderedContent = that.render(content, that.create_context(value),
+ partials, true);
+ } else if (typeof value == "function") {
+ // higher order section
+ renderedContent = value.call(context, content, function (text) {
+ return that.render(text, context, partials, true);
+ });
+ } else if (value) { // boolean section
+ renderedContent = that.render(content, context, partials, true);
+ } else {
+ renderedContent = "";
+ }
+ }
+
+ return renderedBefore + renderedContent + renderedAfter;
+ });
+ },
+
+ /*
+ Replace {{foo}} and friends with values from our view
+ */
+ render_tags: function (template, context, partials, in_recursion) {
+ // tit for tat
+ var that = this;
+
+ var new_regex = function () {
+ return that.getCachedRegex("render_tags", function (otag, ctag) {
+ return new RegExp(otag + "(=|!|>|&|\\{|%)?([^#\\^]+?)\\1?" + ctag + "+", "g");
+ });
+ };
+
+ var regex = new_regex();
+ var tag_replace_callback = function (match, operator, name) {
+ switch(operator) {
+ case "!": // ignore comments
+ return "";
+ case "=": // set new delimiters, rebuild the replace regexp
+ that.set_delimiters(name);
+ regex = new_regex();
+ return "";
+ case ">": // render partial
+ return that.render_partial(name, context, partials);
+ case "{": // the triple mustache is unescaped
+ case "&": // & operator is an alternative unescape method
+ return that.find(name, context);
+ default: // escape the value
+ return escapeHTML(that.find(name, context));
+ }
+ };
+ var lines = template.split("\n");
+ for(var i = 0; i < lines.length; i++) {
+ lines[i] = lines[i].replace(regex, tag_replace_callback, this);
+ if (!in_recursion) {
+ this.send(lines[i]);
+ }
+ }
+
+ if (in_recursion) {
+ return lines.join("\n");
+ }
+ },
+
+ set_delimiters: function (delimiters) {
+ var dels = delimiters.split(" ");
+ this.otag = this.escape_regex(dels[0]);
+ this.ctag = this.escape_regex(dels[1]);
+ },
+
+ escape_regex: function (text) {
+ // thank you Simon Willison
+ if (!arguments.callee.sRE) {
+ var specials = [
+ '/', '.', '*', '+', '?', '|',
+ '(', ')', '[', ']', '{', '}', '\\'
+ ];
+ arguments.callee.sRE = new RegExp(
+ '(\\' + specials.join('|\\') + ')', 'g'
+ );
+ }
+ return text.replace(arguments.callee.sRE, '\\$1');
+ },
+
+ /*
+ find `name` in current `context`. That is find me a value
+ from the view object
+ */
+ find: function (name, context) {
+ name = trim(name);
+
+ // Checks whether a value is thruthy or false or 0
+ function is_kinda_truthy(bool) {
+ return bool === false || bool === 0 || bool;
+ }
+
+ var value;
+
+ // check for dot notation eg. foo.bar
+ if (name.match(/([a-z_]+)\./ig)) {
+ var childValue = this.walk_context(name, context);
+ if (is_kinda_truthy(childValue)) {
+ value = childValue;
+ }
+ } else {
+ if (is_kinda_truthy(context[name])) {
+ value = context[name];
+ } else if (is_kinda_truthy(this.context[name])) {
+ value = this.context[name];
+ }
+ }
+
+ if (typeof value == "function") {
+ return value.apply(context);
+ }
+ if (value !== undefined) {
+ return value;
+ }
+ // silently ignore unkown variables
+ return "";
+ },
+
+ walk_context: function (name, context) {
+ var path = name.split('.');
+ // if the var doesn't exist in current context, check the top level context
+ var value_context = (context[path[0]] != undefined) ? context : this.context;
+ var value = value_context[path.shift()];
+ while (value != undefined && path.length > 0) {
+ value_context = value;
+ value = value[path.shift()];
+ }
+ // if the value is a function, call it, binding the correct context
+ if (typeof value == "function") {
+ return value.apply(value_context);
+ }
+ return value;
+ },
+
+ // Utility methods
+
+ /* includes tag */
+ includes: function (needle, haystack) {
+ return haystack.indexOf(this.otag + needle) != -1;
+ },
+
+ // by @langalex, support for arrays of strings
+ create_context: function (_context) {
+ if (this.is_object(_context)) {
+ return _context;
+ } else {
+ var iterator = ".";
+ if (this.pragmas["IMPLICIT-ITERATOR"]) {
+ iterator = this.pragmas["IMPLICIT-ITERATOR"].iterator;
+ }
+ var ctx = {};
+ ctx[iterator] = _context;
+ return ctx;
+ }
+ },
+
+ is_object: function (a) {
+ return a && typeof a == "object";
+ },
+
+ /*
+ Why, why, why? Because IE. Cry, cry cry.
+ */
+ map: function (array, fn) {
+ if (typeof array.map == "function") {
+ return array.map(fn);
+ } else {
+ var r = [];
+ var l = array.length;
+ for(var i = 0; i < l; i++) {
+ r.push(fn(array[i]));
+ }
+ return r;
+ }
+ },
+
+ getCachedRegex: function (name, generator) {
+ var byOtag = regexCache[this.otag];
+ if (!byOtag) {
+ byOtag = regexCache[this.otag] = {};
+ }
+
+ var byCtag = byOtag[this.ctag];
+ if (!byCtag) {
+ byCtag = byOtag[this.ctag] = {};
+ }
+
+ var regex = byCtag[name];
+ if (!regex) {
+ regex = byCtag[name] = generator(this.otag, this.ctag);
+ }
+
+ return regex;
+ }
+ };
+
+ return({
+ name: "mustache.js",
+ version: "0.4.2",
+
+ /*
+ Turns a template and view into HTML
+ */
+ to_html: function (template, view, partials, send_fun) {
+ var renderer = new Renderer();
+ if (send_fun) {
+ renderer.send = send_fun;
+ }
+ renderer.render(template, view || {}, partials);
+ if (!send_fun) {
+ return renderer.buffer.join("\n");
+ }
+ }
+ });
+}();
View
6 mustache.min.js
@@ -0,0 +1,6 @@
+/*! Version: 0.4.2 */
+/*
+ mustache.js — Logic-less templates in JavaScript
+
+ See http://mustache.github.com/ for more info.
+*/var Mustache=function(){function g(a){return String(a).replace(/&(?!#?\w+;)|[<>"']/g,function(a){return f[a]||a})}var a=Object.prototype.toString;Array.isArray=Array.isArray||function(b){return a.call(b)=="[object Array]"};var b=String.prototype.trim,c;if(b)c=function(a){return a==null?"":b.call(a)};else{var d,e;/\S/.test(" ")?(d=/^[\s\xA0]+/,e=/[\s\xA0]+$/):(d=/^\s+/,e=/\s+$/),c=function(a){return a==null?"":a.toString().replace(d,"").replace(e,"")}}var f={"&":"&amp;","<":"&lt;",">":"&gt;",'"':"&quot;","'":"&#39;"},h={},i=function(){};return i.prototype={otag:"{{",ctag:"}}",pragmas:{},buffer:[],pragmas_implemented:{"IMPLICIT-ITERATOR":!0},context:{},render:function(a,b,c,d){d||(this.context=b,this.buffer=[]);if(!this.includes("",a)){if(d)return a;this.send(a);return}a=this.render_pragmas(a);var e=this.render_section(a,b,c);e===!1&&(e=this.render_tags(a,b,c,d));if(d)return e;this.sendLines(e)},send:function(a){a!==""&&this.buffer.push(a)},sendLines:function(a){if(a){var b=a.split("\n");for(var c=0;c<b.length;c++)this.send(b[c])}},render_pragmas:function(a){if(!this.includes("%",a))return a;var b=this,c=this.getCachedRegex("render_pragmas",function(a,b){return new RegExp(a+"%([\\w-]+) ?([\\w]+=[\\w]+)?"+b,"g")});return a.replace(c,function(a,c,d){if(!b.pragmas_implemented[c])throw{message:"This implementation of mustache doesn't understand the '"+c+"' pragma"};b.pragmas[c]={};if(d){var e=d.split("=");b.pragmas[c][e[0]]=e[1]}return""})},render_partial:function(a,b,d){a=c(a);if(!d||d[a]===undefined)throw{message:"unknown_partial '"+a+"'"};return!b||typeof b[a]!="object"?this.render(d[a],b,d,!0):this.render(d[a],b[a],d,!0)},render_section:function(a,b,c){if(!this.includes("#",a)&&!this.includes("^",a))return!1;var d=this,e=this.getCachedRegex("render_section",function(a,b){return new RegExp("^([\\s\\S]*?)"+a+"(\\^|\\#)\\s*(.+?)\\s*"+b+"\n*([\\s\\S]*?)"+a+"\\/\\s*\\3\\s*"+b+"\\s*([\\s\\S]*)$","g")});return a.replace(e,function(a,e,f,g,h,i){var j=e?d.render_tags(e,b,c,!0):"",k=i?d.render(i,b,c,!0):"",l,m=d.find(g,b);return f==="^"?!m||Array.isArray(m)&&m.length===0?l=d.render(h,b,c,!0):l="":f==="#"&&(Array.isArray(m)?l=d.map(m,function(a){return d.render(h,d.create_context(a),c,!0)}).join(""):d.is_object(m)?l=d.render(h,d.create_context(m),c,!0):typeof m=="function"?l=m.call(b,h,function(a){return d.render(a,b,c,!0)}):m?l=d.render(h,b,c,!0):l=""),j+l+k})},render_tags:function(a,b,c,d){var e=this,f=function(){return e.getCachedRegex("render_tags",function(a,b){return new RegExp(a+"(=|!|>|&|\\{|%)?([^#\\^]+?)\\1?"+b+"+","g")})},h=f(),i=function(a,d,i){switch(d){case"!":return"";case"=":return e.set_delimiters(i),h=f(),"";case">":return e.render_partial(i,b,c);case"{":case"&":return e.find(i,b);default:return g(e.find(i,b))}},j=a.split("\n");for(var k=0;k<j.length;k++)j[k]=j[k].replace(h,i,this),d||this.send(j[k]);if(d)return j.join("\n")},set_delimiters:function(a){var b=a.split(" ");this.otag=this.escape_regex(b[0]),this.ctag=this.escape_regex(b[1])},escape_regex:function(a){if(!arguments.callee.sRE){var b=["/",".","*","+","?","|","(",")","[","]","{","}","\\"];arguments.callee.sRE=new RegExp("(\\"+b.join("|\\")+")","g")}return a.replace(arguments.callee.sRE,"\\$1")},find:function(a,b){function d(a){return a===!1||a===0||a}a=c(a);var e;if(a.match(/([a-z_]+)\./ig)){var f=this.walk_context(a,b);d(f)&&(e=f)}else d(b[a])?e=b[a]:d(this.context[a])&&(e=this.context[a]);return typeof e=="function"?e.apply(b):e!==undefined?e:""},walk_context:function(a,b){var c=a.split("."),d=b[c[0]]!=undefined?b:this.context,e=d[c.shift()];while(e!=undefined&&c.length>0)d=e,e=e[c.shift()];return typeof e=="function"?e.apply(d):e},includes:function(a,b){return b.indexOf(this.otag+a)!=-1},create_context:function(a){if(this.is_object(a))return a;var b=".";this.pragmas["IMPLICIT-ITERATOR"]&&(b=this.pragmas["IMPLICIT-ITERATOR"].iterator);var c={};return c[b]=a,c},is_object:function(a){return a&&typeof a=="object"},map:function(a,b){if(typeof a.map=="function")return a.map(b);var c=[],d=a.length;for(var e=0;e<d;e++)c.push(b(a[e]));return c},getCachedRegex:function(a,b){var c=h[this.otag];c||(c=h[this.otag]={});var d=c[this.ctag];d||(d=c[this.ctag]={});var e=d[a];return e||(e=d[a]=b(this.otag,this.ctag)),e}},{name:"mustache.js",version:"0.4.2",to_html:function(a,b,c,d){var e=new i;d&&(e.send=d),e.render(a,b||{},c);if(!d)return e.buffer.join("\n")}}}();
View
3  spec/_files/ampersand_escape.js
@@ -0,0 +1,3 @@
+var ampersand_escape = {
+ message: "Some <code>"
+};
View
1  spec/_files/ampersand_escape.mustache
@@ -0,0 +1 @@
+{{&message}}
View
1  spec/_files/ampersand_escape.txt
@@ -0,0 +1 @@
+Some <code>
View
1  spec/_files/apostrophe.js
@@ -0,0 +1 @@
+var apostrophe = {'apos': "'", 'control':'X'};
View
1  spec/_files/apostrophe.mustache
@@ -0,0 +1 @@
+{{apos}}{{control}}
View
1  spec/_files/apostrophe.txt
@@ -0,0 +1 @@
+&#39;X
View
1  spec/_files/array_of_partials_implicit_partial.2.mustache
@@ -0,0 +1 @@
+{{.}}
View
3  spec/_files/array_of_partials_implicit_partial.js
@@ -0,0 +1,3 @@
+var partial_context = {
+ numbers: ['1', '2', '3', '4']
+};
View
4 spec/_files/array_of_partials_implicit_partial.mustache
@@ -0,0 +1,4 @@
+Here is some stuff!
+{{#numbers}}
+{{>partial}}
+{{/numbers}}
View
5 spec/_files/array_of_partials_implicit_partial.txt
@@ -0,0 +1,5 @@
+Here is some stuff!
+1
+2
+3
+4
View
1  spec/_files/array_of_partials_partial.2.mustache
@@ -0,0 +1 @@
+{{i}}
View
3  spec/_files/array_of_partials_partial.js
@@ -0,0 +1,3 @@
+var partial_context = {
+ numbers: [{i: '1'}, {i: '2'}, {i: '3'}, {i: '4'}]
+};
View
4 spec/_files/array_of_partials_partial.mustache
@@ -0,0 +1,4 @@
+Here is some stuff!
+{{#numbers}}
+{{>partial}}
+{{/numbers}}
View
5 spec/_files/array_of_partials_partial.txt
@@ -0,0 +1,5 @@
+Here is some stuff!
+1
+2
+3
+4
View
1  spec/_files/array_of_strings.js
@@ -0,0 +1 @@
+var array_of_strings = {array_of_strings: ['hello', 'world']};
View
1  spec/_files/array_of_strings.mustache
@@ -0,0 +1 @@
+{{#array_of_strings}}{{.}} {{/array_of_strings}}
View
1  spec/_files/array_of_strings.txt
@@ -0,0 +1 @@
+hello world
View
1  spec/_files/array_of_strings_options.js
@@ -0,0 +1 @@
+var array_of_strings_options = {array_of_strings_options: ['hello', 'world']};
View
2  spec/_files/array_of_strings_options.mustache
@@ -0,0 +1,2 @@
+{{%IMPLICIT-ITERATOR iterator=rob}}
+{{#array_of_strings_options}}{{rob}} {{/array_of_strings_options}}
View
1  spec/_files/array_of_strings_options.txt
@@ -0,0 +1 @@
+hello world
View
4 spec/_files/array_partial.2.mustache
@@ -0,0 +1,4 @@
+Here's a non-sense array of values
+{{#array}}
+ {{.}}
+{{/array}}
View
5 spec/_files/array_partial.js
@@ -0,0 +1,5 @@
+var partial_context = {
+ partial: {
+ array: ['1', '2', '3', '4']
+ }
+};
View
1  spec/_files/array_partial.mustache
@@ -0,0 +1 @@
+{{>partial}}
View
6 spec/_files/array_partial.txt
@@ -0,0 +1,6 @@
+Here's a non-sense array of values
+ 1
+ 2
+ 3
+ 4
+
View
3  spec/_files/bug_11_eating_whitespace.js
@@ -0,0 +1,3 @@
+var bug_11_eating_whitespace = {
+ tag: "yo"
+};
View
1  spec/_files/bug_11_eating_whitespace.mustache
@@ -0,0 +1 @@
+{{tag}} foo
View
1  spec/_files/bug_11_eating_whitespace.txt
@@ -0,0 +1 @@
+yo foo
View
5 spec/_files/comments.js
@@ -0,0 +1,5 @@
+var comments = {
+ title: function() {
+ return "A Comedy of Errors";
+ }
+};
View
1  spec/_files/comments.mustache
@@ -0,0 +1 @@
+<h1>{{title}}{{! just something interesting... or not... }}</h1>
View
1  spec/_files/comments.txt
@@ -0,0 +1 @@
+<h1>A Comedy of Errors</h1>
View
19 spec/_files/complex.js
@@ -0,0 +1,19 @@
+var complex = {
+ header: function() {
+ return "Colors";
+ },
+ item: [
+ {name: "red", current: true, url: "#Red"},
+ {name: "green", current: false, url: "#Green"},
+ {name: "blue", current: false, url: "#Blue"}
+ ],
+ link: function() {
+ return this["current"] !== true;
+ },
+ list: function() {
+ return this.item.length !== 0;
+ },
+ empty: function() {
+ return this.item.length === 0;
+ }
+};
View
16 spec/_files/complex.mustache
@@ -0,0 +1,16 @@
+<h1>{{header}}</h1>
+{{#list}}
+ <ul>
+ {{#item}}
+ {{#current}}
+ <li><strong>{{name}}</strong></li>
+ {{/current}}
+ {{#link}}
+ <li><a href="{{url}}">{{name}}</a></li>
+ {{/link}}
+ {{/item}}
+ </ul>
+{{/list}}
+{{#empty}}
+ <p>The list is empty.</p>
+{{/empty}}
View
6 spec/_files/complex.txt
@@ -0,0 +1,6 @@
+<h1>Colors</h1>
+ <ul>
+ <li><strong>red</strong></li>
+ <li><a href="#Green">green</a></li>
+ <li><a href="#Blue">blue</a></li>
+ </ul>
View
6 spec/_files/delimiters.js
@@ -0,0 +1,6 @@
+var delimiters = {
+ first: "It worked the first time.",
+ second: "And it worked the second time.",
+ third: "Then, surprisingly, it worked the third time.",
+ fourth: "Fourth time also fine!."
+}
View
7 spec/_files/delimiters.mustache
@@ -0,0 +1,7 @@
+{{=<% %>=}}*
+<% first %>
+* <% second %>
+<%=| |=%>
+* | third |
+|={{ }}=|
+* {{ fourth }}
View
5 spec/_files/delimiters.txt
@@ -0,0 +1,5 @@
+*
+It worked the first time.
+* And it worked the second time.
+* Then, surprisingly, it worked the third time.
+* Fourth time also fine!.
View
23 spec/_files/dot_notation.js
@@ -0,0 +1,23 @@
+var dot_notation = {
+ name: "A Book",
+ authors: ["John Power", "Jamie Walsh"],
+ price:{
+ value: 200,
+ vat: function() {
+ return this.value * 0.2;
+ },
+ currency: {
+ symbol: '&euro;',
+ name: 'Euro'
+ }
+ },
+ availability:{
+ status: true,
+ text: "In Stock"
+ },
+ // And now, some truthy false values
+ truthy: {
+ zero: 0,
+ notTrue: false
+ }
+};
View
9 spec/_files/dot_notation.mustache
@@ -0,0 +1,9 @@
+<!-- exciting part -->
+<h1>{{name}}</h1>
+<p>Authors: <ul>{{#authors}}<li>{{.}}</li>{{/authors}}</ul></p>
+<p>Price: {{price.currency.symbol}}{{price.value}} {{#price.currency}}{{name}} <b>{{availability.text}}</b>{{/price.currency}}</p>
+<p>VAT: {{price.currency.symbol}}{{price.vat}}</p>
+<!-- boring part -->
+<h2>Test truthy false values:</h2>
+<p>Zero: {{truthy.zero}}</p>
+<p>False: {{truthy.notTrue}}</p>
View
9 spec/_files/dot_notation.txt
@@ -0,0 +1,9 @@
+<!-- exciting part -->
+<h1>A Book</h1>
+<p>Authors: <ul><li>John Power</li><li>Jamie Walsh</li></ul></p>
+<p>Price: &euro;200 Euro <b>In Stock</b></p>
+<p>VAT: &euro;40</p>
+<!-- boring part -->
+<h2>Test truthy false values:</h2>
+<p>Zero: 0</p>
+<p>False: false</p>
View
5 spec/_files/double_render.js
@@ -0,0 +1,5 @@
+var double_render = {
+ foo: true,
+ bar: "{{win}}",
+ win: "FAIL"
+};
View
1  spec/_files/double_render.mustache
@@ -0,0 +1 @@
+{{#foo}}{{bar}}{{/foo}}
View
1  spec/_files/double_render.txt
@@ -0,0 +1 @@
+{{win}}
View
1  spec/_files/empty_partial.2.mustache
@@ -0,0 +1 @@
+yo
View
3  spec/_files/empty_partial.js
@@ -0,0 +1,3 @@
+var partial_context = {
+ foo: 1
+};
View
2  spec/_files/empty_partial.mustache
@@ -0,0 +1,2 @@
+hey {{foo}}
+{{>partial}}
View
2  spec/_files/empty_partial.txt
@@ -0,0 +1,2 @@
+hey 1
+yo
View
1  spec/_files/empty_sections.js
@@ -0,0 +1 @@
+var empty_sections = {};
View
1  spec/_files/empty_sections.mustache
@@ -0,0 +1 @@
+{{#foo}}{{/foo}}foo{{#bar}}{{/bar}}
View
1  spec/_files/empty_sections.txt
@@ -0,0 +1 @@
+foo
View
1  spec/_files/empty_template.js
@@ -0,0 +1 @@
+var empty_template = {};
View
1  spec/_files/empty_template.mustache
@@ -0,0 +1 @@
+<html><head></head><body><h1>Test</h1></body></html>
View
1  spec/_files/empty_template.txt
@@ -0,0 +1 @@
+<html><head></head><body><h1>Test</h1></body></html>
View
1  spec/_files/error_not_found.js
@@ -0,0 +1 @@
+var error_not_found = {bar: 2};
View
1  spec/_files/error_not_found.mustache
@@ -0,0 +1 @@
+{{foo}}
View
1  spec/_files/error_not_found.txt
@@ -0,0 +1 @@
+
View
7 spec/_files/escaped.js
@@ -0,0 +1,7 @@
+var escaped = {
+ title: function() {
+ return "Bear > Shark";
+ },
+ subtitle: 'Rock &#39;n Roll',
+ entities: "&quot;"
+};
View
3  spec/_files/escaped.mustache
@@ -0,0 +1,3 @@
+<h1>{{title}}</h1>
+<h2>{{subtitle}}</h2>
+But not {{entities}}.
View
3  spec/_files/escaped.txt
@@ -0,0 +1,3 @@
+<h1>Bear &gt; Shark</h1>
+<h2>Rock &#39;n Roll</h2>
+But not &quot;.
View
9 spec/_files/higher_order_sections.js
@@ -0,0 +1,9 @@
+var higher_order_sections = {
+ "name": "Tater",
+ "helper": "To tinker?",
+ "bolder": function() {
+ return function(text, render) {
+ return "<b>" + render(text) + '</b> ' + this.helper;
+ }
+ }
+}
View
1  spec/_files/higher_order_sections.mustache
@@ -0,0 +1 @@
+{{#bolder}}Hi {{name}}.{{/bolder}}
View
1  spec/_files/higher_order_sections.txt
@@ -0,0 +1 @@
+<b>Hi Tater.</b> To tinker?
View
3  spec/_files/inverted_section.js
@@ -0,0 +1,3 @@
+var inverted_section = {
+ "repo": []
+}
View
2  spec/_files/inverted_section.mustache
@@ -0,0 +1,2 @@
+{{#repo}}<b>{{name}}</b>{{/repo}}
+{{^repo}}No repos :({{/repo}}
View
1  spec/_files/inverted_section.txt
@@ -0,0 +1 @@
+No repos :(
View
5 spec/_files/keys_with_questionmarks.js
@@ -0,0 +1,5 @@
+var keys_with_questionmarks = {
+ "person?": {
+ name: "Jon"
+ }
+}
View
3  spec/_files/keys_with_questionmarks.mustache
@@ -0,0 +1,3 @@
+{{#person?}}
+ Hi {{name}}!
+{{/person?}}
View
1  spec/_files/keys_with_questionmarks.txt
@@ -0,0 +1 @@
+ Hi Jon!
View
7 spec/_files/nesting.js
@@ -0,0 +1,7 @@
+var nesting = {
+ foo: [
+ {a: {b: 1}},
+ {a: {b: 2}},
+ {a: {b: 3}}
+ ]
+};
View
5 spec/_files/nesting.mustache
@@ -0,0 +1,5 @@
+{{#foo}}
+ {{#a}}
+ {{b}}
+ {{/a}}
+{{/foo}}
View
4 spec/_files/nesting.txt
@@ -0,0 +1,4 @@
+ 1
+ 2
+ 3
+
View
9 spec/_files/null_string.js
@@ -0,0 +1,9 @@
+var null_string = {
+ name: "Elise",
+ glytch: true,
+ binary: false,
+ value: null,
+ numeric: function() {
+ return NaN;
+ }
+};
View
5 spec/_files/null_string.mustache
@@ -0,0 +1,5 @@
+Hello {{name}}
+glytch {{glytch}}
+binary {{binary}}
+value {{value}}
+numeric {{numeric}}
View
5 spec/_files/null_string.txt
@@ -0,0 +1,5 @@
+Hello Elise
+glytch true
+binary false
+value
+numeric NaN
View
4 spec/_files/partial_recursion.2.mustache
@@ -0,0 +1,4 @@
+{{name}}
+{{#children}}
+{{>partial}}
+{{/children}}
View
11 spec/_files/partial_recursion.js
@@ -0,0 +1,11 @@
+var partial_context = {
+ name: '1',
+ kids: [
+ {
+ name: '1.1',
+ children: [
+ {name: '1.1.1'}
+ ]
+ }
+ ]
+};
View
4 spec/_files/partial_recursion.mustache
@@ -0,0 +1,4 @@
+{{name}}
+{{#kids}}
+{{>partial}}
+{{/kids}}
View
3  spec/_files/partial_recursion.txt
@@ -0,0 +1,3 @@
+1
+1.1
+1.1.1
View
8 spec/_files/recursion_with_same_names.js
@@ -0,0 +1,8 @@
+var recursion_with_same_names = {
+ name: 'name',
+ description: 'desc',
+ terms: [
+ {name: 't1', index: 0},
+ {name: 't2', index: 1},
+ ]
+};
View
7 spec/_files/recursion_with_same_names.mustache
@@ -0,0 +1,7 @@
+{{ name }}
+{{ description }}
+
+{{#terms}}
+ {{name}}
+ {{index}}
+{{/terms}}
View
6 spec/_files/recursion_with_same_names.txt
@@ -0,0 +1,6 @@
+name
+desc
+ t1
+ 0
+ t2
+ 1
View
6 spec/_files/reuse_of_enumerables.js
@@ -0,0 +1,6 @@
+var reuse_of_enumerables = {
+ terms: [
+ {name: 't1', index: 0},
+ {name: 't2', index: 1},
+ ]
+};
View
8 spec/_files/reuse_of_enumerables.mustache
@@ -0,0 +1,8 @@
+{{#terms}}
+ {{name}}
+ {{index}}
+{{/terms}}
+{{#terms}}
+ {{name}}
+ {{index}}
+{{/terms}}
View
8 spec/_files/reuse_of_enumerables.txt
@@ -0,0 +1,8 @@
+ t1
+ 0
+ t2
+ 1
+ t1
+ 0
+ t2
+ 1
View
7 spec/_files/section_as_context.js
@@ -0,0 +1,7 @@
+var section_as_context = {
+ a_object: {
+ title: 'this is an object',
+ description: 'one of its attributes is a list',
+ a_list: [{label: 'listitem1'}, {label: 'listitem2'}]
+ }
+};
View
9 spec/_files/section_as_context.mustache
@@ -0,0 +1,9 @@
+{{#a_object}}
+ <h1>{{title}}</h1>
+ <p>{{description}}</p>
+ <ul>
+ {{#a_list}}
+ <li>{{label}}</li>
+ {{/a_list}}
+ </ul>
+{{/a_object}}
View
6 spec/_files/section_as_context.txt
@@ -0,0 +1,6 @@
+ <h1>this is an object</h1>
+ <p>one of its attributes is a list</p>
+ <ul>
+ <li>listitem1</li>
+ <li>listitem2</li>
+ </ul>
View
8 spec/_files/simple.js
@@ -0,0 +1,8 @@
+var simple = {
+ name: "Chris",
+ value: 10000,
+ taxed_value: function() {
+ return this.value - (this.value * 0.4);
+ },
+ in_ca: true
+};
View
5 spec/_files/simple.mustache
@@ -0,0 +1,5 @@
+Hello {{name}}
+You have just won ${{value}}!
+{{#in_ca}}
+Well, ${{ taxed_value }}, after taxes.
+{{/in_ca}}
View
3  spec/_files/simple.txt
@@ -0,0 +1,3 @@
+Hello Chris
+You have just won $10000!
+Well, $6000, after taxes.
View
1  spec/_files/template_partial.2.mustache
@@ -0,0 +1 @@
+Again, {{again}}!
View
8 spec/_files/template_partial.js
@@ -0,0 +1,8 @@
+var partial_context = {
+ title: function() {
+ return "Welcome";
+ },
+ partial: {
+ again: "Goodbye"
+ }
+}
View
2  spec/_files/template_partial.mustache
@@ -0,0 +1,2 @@
+<h1>{{title}}</h1>
+{{>partial}}
View
2  spec/_files/template_partial.txt
@@ -0,0 +1,2 @@
+<h1>Welcome</h1>
+Again, Goodbye!
View
4 spec/_files/two_in_a_row.js
@@ -0,0 +1,4 @@
+var two_in_a_row = {
+ name: "Joe",
+ greeting: "Welcome"
+};
View
1  spec/_files/two_in_a_row.mustache
@@ -0,0 +1 @@
+{{greeting}}, {{name}}!
View
1  spec/_files/two_in_a_row.txt
@@ -0,0 +1 @@
+Welcome, Joe!
View
1  spec/_files/two_sections.js
@@ -0,0 +1 @@
+var two_sections = {};
View
4 spec/_files/two_sections.mustache
@@ -0,0 +1,4 @@
+{{#foo}}
+{{/foo}}
+{{#bar}}
+{{/bar}}
View
1  spec/_files/two_sections.txt
@@ -0,0 +1 @@
+
View
5 spec/_files/unescaped.js
@@ -0,0 +1,5 @@
+var unescaped = {
+ title: function() {
+ return "Bear > Shark";
+ }
+};
View
1  spec/_files/unescaped.mustache
@@ -0,0 +1 @@
+<h1>{{{title}}}</h1>
View
1  spec/_files/unescaped.txt
@@ -0,0 +1 @@
+<h1>Bear > Shark</h1>
View
1  spec/_files/unknown_pragma.js
@@ -0,0 +1 @@
+var unknown_pragma = {};
View
1  spec/_files/unknown_pragma.mustache
@@ -0,0 +1 @@
+{{%I-HAVE-THE-GREATEST-MUSTACHE}}
View
1  spec/_files/unknown_pragma.txt
@@ -0,0 +1 @@
+ERROR: This implementation of mustache doesn't understand the 'I-HAVE-THE-GREATEST-MUSTACHE' pragma
View
5 spec/_files/view_partial.2.mustache
@@ -0,0 +1,5 @@
+Hello {{name}}
+You have just won ${{value}}!
+{{#in_ca}}
+Well, ${{ taxed_value }}, after taxes.
+{{/in_ca}}
View
19 spec/_files/view_partial.js
@@ -0,0 +1,19 @@
+var partial_context = {
+ greeting: function() {
+ return "Welcome";
+ },
+
+ farewell: function() {
+ return "Fair enough, right?";
+ },
+
+ partial: {
+ name: "Chris",
+ value: 10000,
+ taxed_value: function() {
+ return this.value - (this.value * 0.4);
+ },
+ in_ca: true
+ }
+};
+
View
3  spec/_files/view_partial.mustache
@@ -0,0 +1,3 @@
+<h1>{{greeting}}</h1>
+{{>partial}}
+<h3>{{farewell}}</h3>
View
6 spec/_files/view_partial.txt
@@ -0,0 +1,6 @@
+<h1>Welcome</h1>
+Hello Chris
+You have just won $10000!
+Well, $6000, after taxes.
+
+<h3>Fair enough, right?</h3>
View
5 spec/_files/whitespace_partial.2.mustache
@@ -0,0 +1,5 @@
+Hello {{ name}}
+You have just won ${{value }}!
+{{# in_ca }}
+Well, ${{ taxed_value }}, after taxes.
+{{/ in_ca }}
View
19 spec/_files/whitespace_partial.js
@@ -0,0 +1,19 @@
+var partial_context = {
+ greeting: function() {
+ return "Welcome";
+ },
+
+ farewell: function() {
+ return "Fair enough, right?";
+ },
+
+ partial: {
+ name: "Chris",
+ value: 10000,
+ taxed_value: function() {
+ return this.value - (this.value * 0.4);
+ },
+ in_ca: true
+ }
+};
+
View
3  spec/_files/whitespace_partial.mustache
@@ -0,0 +1,3 @@
+<h1>{{ greeting }}</h1>
+{{> partial }}
+<h3>{{ farewell }}</h3>
View
6 spec/_files/whitespace_partial.txt
@@ -0,0 +1,6 @@
+<h1>Welcome</h1>
+Hello Chris
+You have just won $10000!
+Well, $6000, after taxes.
+
+<h3>Fair enough, right?</h3>
View
276 spec/mustache_spec.rb
@@ -0,0 +1,276 @@
+require 'rubygems'
+require 'json'
+
+ROOT = File.expand_path('../..', __FILE__)
+SPEC = File.join(ROOT, 'spec')
+FILES = File.join(SPEC, '_files')
+
+MUSTACHE = File.read(File.join(ROOT, "mustache.js"))
+
+TESTS = Dir.glob(File.join(FILES, '*.js')).map do |name|
+ File.basename name, '.js'
+end
+
+PARTIALS = TESTS.select {|t| t.include? "partial" }
+NON_PARTIALS = TESTS.select {|t| not t.include? "partial" }
+
+NODE_PATH = `which node`.strip
+JS_PATH = `which js`.strip
+JSC_PATH = "/System/Library/Frameworks/JavaScriptCore.framework/Versions/A/Resources/jsc"
+RHINO_JAR = "org.mozilla.javascript.tools.shell.Main"
+
+def load_test(name, is_partial=false)
+ view = File.read(File.join(FILES, "#{name}.js"))
+ template = File.read(File.join(FILES, "#{name}.mustache")).to_json
+ expect = File.read(File.join(FILES, "#{name}.txt"))
+
+ test = [view, template, expect]
+
+ if is_partial
+ test << File.read(File.join(FILES, "#{name}.2.mustache")).to_json
+ end
+
+ test
+end
+
+def run_js(runner, js)
+ cmd = case runner
+ when :spidermonkey
+ JS_PATH
+ when :jsc
+ JSC_PATH
+ when :rhino
+ "java #{RHINO_JAR}"
+ when :node
+ NODE_PATH
+ end
+
+ runner_file = "runner.js"
+ File.open(runner_file, 'w') {|file| file.write(js) }
+ `#{cmd} #{runner_file}`
+ensure
+ FileUtils.rm_r(runner_file)
+end
+
+$engines_run = 0
+
+describe "mustache" do
+ shared_examples_for "mustache rendering" do
+ before(:all) do
+ $engines_run += 1
+ end
+
+ it "should return the same when invoked multiple times" do
+ js = <<-JS
+ #{@boilerplate}
+ Mustache.to_html("x")
+ print(Mustache.to_html("x"));
+ JS
+
+ run_js(@runner, js).should == "x\n"
+ end
+
+ it "should clear the context after each run" do
+ js = <<-JS
+ #{@boilerplate}
+ Mustache.to_html("{{#list}}{{x}}{{/list}}", {list: [{x: 1}]})
+ try {
+ print(Mustache.to_html("{{#list}}{{x}}{{/list}}", {list: [{}]}));
+ } catch(e) {
+ print('ERROR: ' + e.message);
+ }
+ JS
+
+ run_js(@runner, js).should == "\n"
+ end
+
+ NON_PARTIALS.each do |test|
+ describe test do
+ it "should generate the correct html" do
+ view, template, expect = load_test(test)
+
+ js = <<-JS
+ try {
+ #{@boilerplate}
+ #{view}
+ var template = #{template};
+ var result = Mustache.to_html(template, #{test});
+ print(result);
+ } catch(e) {
+ print('ERROR: ' + e.message);
+ }
+ JS
+
+ run_js(@runner, js).should == expect
+ end
+
+ it "should sendFun the correct html" do
+ view, template, expect = load_test(test)
+
+ js = <<-JS
+ try {
+ #{@boilerplate}
+ #{view}
+ var chunks = [];
+ var sendFun = function(chunk) {
+ if (chunk != "") {
+ chunks.push(chunk);
+ }
+ }
+ var template = #{template};
+ Mustache.to_html(template, #{test}, null, sendFun);
+ print(chunks.join("\\n"));
+ } catch(e) {
+ print('ERROR: ' + e.message);
+ }
+ JS
+
+ run_js(@runner, js).strip.should == expect.strip
+ end
+ end
+ end
+
+ PARTIALS.each do |test|
+ describe test do
+ it "should generate the correct html" do
+ view, template, expect, partial = load_test(test, true)
+
+ js = <<-JS
+ try {
+ #{@boilerplate}
+ #{view}
+ var template = #{template};
+ var partials = {"partial": #{partial}};
+ var result = Mustache.to_html(template, partial_context, partials);
+ print(result);
+ } catch(e) {
+ print('ERROR: ' + e.message);
+ }
+ JS
+
+ run_js(@runner, js).should == expect
+ end
+
+ it "should sendFun the correct html" do
+ view, template, expect, partial = load_test(test, true)
+
+ js = <<-JS
+ try {
+ #{@boilerplate}
+ #{view};
+ var template = #{template};
+ var partials = {"partial": #{partial}};
+ var chunks = [];
+ var sendFun = function(chunk) {
+ if (chunk != "") {
+ chunks.push(chunk);
+ }
+ }
+ Mustache.to_html(template, partial_context, partials, sendFun);
+ print(chunks.join("\\n"));
+ } catch(e) {
+ print('ERROR: ' + e.message);
+ }
+ JS
+
+ run_js(@runner, js).strip.should == expect.strip
+ end
+ end
+ end
+ end
+
+ context "running in node" do
+ if File.exist?(NODE_PATH)
+ before(:all) do
+ $stdout.write "Testing in node "
+ @runner = :node
+ @boilerplate = MUSTACHE.dup
+ @boilerplate << <<-JS
+ function print(message) {
+ console.log(message);
+ }
+ JS
+ end
+
+ after(:all) do
+ puts " Done!"
+ end
+
+ it_should_behave_like "mustache rendering"
+ else
+ puts "Skipping tests in node (node not found)"
+ end
+ end
+
+ context "running in SpiderMonkey (Mozilla, Firefox)" do
+ if File.exist?(JS_PATH)
+ before(:all) do
+ $stdout.write "Testing in SpiderMonkey "
+ @runner = :spidermonkey
+ @boilerplate = MUSTACHE.dup
+ end
+
+ after(:all) do
+ puts " Done!"
+ end
+
+ it_should_behave_like "mustache rendering"
+ else
+ puts "Skipping tests in SpiderMonkey (js not found)"
+ end
+ end
+
+ context "running in JavaScriptCore (WebKit, Safari)" do
+ if File.exist?(JSC_PATH)
+ before(:all) do
+ $stdout.write "Testing in JavaScriptCore "
+ @runner = :jsc
+ @boilerplate = MUSTACHE.dup
+ end
+
+ after(:all) do
+ puts " Done!"
+ end
+
+ it_should_behave_like "mustache rendering"
+ else
+ puts "Skipping tests in JavaScriptCore (jsc not found)"
+ end
+ end
+
+ context "running in Rhino (Mozilla, Java)" do
+ if `java #{RHINO_JAR} 'foo' 2>&1` !~ /ClassNotFoundException/
+ before(:all) do
+ $stdout.write "Testing in Rhino "
+ @runner = :rhino
+ @boilerplate = MUSTACHE.dup
+ end
+
+ after(:all) do
+ puts " Done!"
+ end
+
+ it_should_behave_like "mustache rendering"
+ else
+ puts "Skipping tests in Rhino (JAR #{RHINO_JAR} was not found)"
+ end
+ end
+
+ context "suite" do
+ before(:each) do
+ $stdout.write "Verifying that we ran at the tests in at least one engine ... "
+ end
+
+ after(:each) do
+ if @exception.nil?
+ puts "OK"
+ else
+ puts "ERROR!"
+ end
+ end
+
+ it "should have run at least one time" do
+ $engines_run.should > 0
+ end
+ end
+end
View
8 wrappers/commonjs/mustache.js.tpl.post
@@ -0,0 +1,8 @@
+if (typeof module !== 'undefined' && module.exports) {
+ exports.name = Mustache.name;
+ exports.version = Mustache.version;
+
+ exports.to_html = function() {
+ return Mustache.to_html.apply(this, arguments);
+ };
+}
View
6 wrappers/commonjs/mustache.js.tpl.pre
@@ -0,0 +1,6 @@
+/*
+ * CommonJS-compatible mustache.js module
+ *
+ * See http://github.com/janl/mustache.js for more info.
+ */
+
View
8 wrappers/commonjs/package.json
@@ -0,0 +1,8 @@
+{
+ "name": "mustache",
+ "author": "http://mustache.github.com/",
+ "description": "Logic-less {{mustache}} templates with JavaScript",
+ "keywords": ["template", "templates", "mustache"],
+ "version": "{{version}}",
+ "main": "./mustache"
+}
View
4 wrappers/dojo/mustache.js.tpl.post
@@ -0,0 +1,4 @@
+
+ dojox.mustache = dojo.hitch(Mustache, "to_html");
+
+})();
View
9 wrappers/dojo/mustache.js.tpl.pre
@@ -0,0 +1,9 @@
+/*
+Shameless port of a shameless port
+@defunkt => @janl => @aq => @voodootikigod
+
+See http://github.com/defunkt/mustache for more info.
+*/
+
+dojo.provide("dojox.mustache._base");
+(function(){
View
6 wrappers/jquery/jquery.mustache.js.tpl.post
@@ -0,0 +1,6 @@
+
+ $.mustache = function(template, view, partials) {
+ return Mustache.to_html(template, view, partials);
+ };
+
+})(jQuery);
View
9 wrappers/jquery/jquery.mustache.js.tpl.pre
@@ -0,0 +1,9 @@
+/*
+Shameless port of a shameless port
+@defunkt => @janl => @aq
+
+See http://github.com/defunkt/mustache for more info.
+*/
+
+;(function($) {
+
View
9 wrappers/qooxdoo/qooxdoo.mustache.js.tpl.post
@@ -0,0 +1,9 @@
+/**
+ * Above is the original mustache code.
+ */
+
+// EXPOSE qooxdoo variant
+qx.bom.Template.version = Mustache.version;
+qx.bom.Template.toHtml = Mustache.to_html;
+
+})();
View
127 wrappers/qooxdoo/qooxdoo.mustache.js.tpl.pre
@@ -0,0 +1,127 @@
+/* ************************************************************************
+
+ qooxdoo - the new era of web development
+
+ http://qooxdoo.org
+
+ Copyright:
+ 2004-2011 1&1 Internet AG, Germany, http://www.1und1.de
+
+ License:
+ LGPL: http://www.gnu.org/licenses/lgpl.html
+ EPL: http://www.eclipse.org/org/documents/epl-v10.php
+ See the LICENSE file in the project's top-level directory for details.
+
+ Authors:
+ * Martin Wittemann (martinwittemann)
+
+ ======================================================================
+
+ This class contains code based on the following work:
+
+ * Mustache.js version 0.4.2
+
+ Code:
+ https://github.com/janl/mustache.js
+
+ Copyright:
+ (c) 2009 Chris Wanstrath (Ruby)
+ (c) 2010 Jan Lehnardt (JavaScript)
+
+ License:
+ MIT: http://www.opensource.org/licenses/mit-license.php
+
+ ----------------------------------------------------------------------
+
+ Copyright (c) 2009 Chris Wanstrath (Ruby)
+ Copyright (c) 2010 Jan Lehnardt (JavaScript)
+
+ Permission is hereby granted, free of charge, to any person obtaining
+ a copy of this software and associated documentation files (the
+ "Software"), to deal in the Software without restriction, including
+ without limitation the rights to use, copy, modify, merge, publish,
+ distribute, sublicense, and/or sell copies of the Software, and to
+ permit persons to whom the Software is furnished to do so, subject to
+ the following conditions:
+
+ The above copyright notice and this permission notice shall be
+ included in all copies or substantial portions of the Software.
+
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+
+************************************************************************ */
+
+/**
+ * The is a template class which can be used for HTML templating. In fact,
+ * this is a wrapper for mustache.js which is a "framework-agnostic way to
+ * render logic-free views".
+ *
+ * Here is a basic example how to use it:
+ * Template:
+ * <pre>
+ * var template = "Hi, my name is {{name}}!";
+ * var view = {name: "qooxdoo"};
+ * qx.bom.Template.toHtml(template, view);
+ * // return "Hi, my name is qooxdoo!"
+ * </pre>
+ *
+ * For further details, please visit the mustache.js documentation here:
+ * https://github.com/janl/mustache.js/blob/master/README.md
+ */
+qx.Class.define("qx.bom.Template", {
+ statics : {
+ /** Contains the mustache.js version. */
+ version: null,
+
+ /**
+ * Original and only template method of mustache.js. For further
+ * documentation, please visit https://github.com/janl/mustache.js
+ *
+ * @signature function(template, view, partials, send_fun)
+ * @param template {String} The String containing the template.
+ * @param view {Object} The object holding the data to render.
+ * @param partials {Object} Object holding parts of a template.
+ * @param send_fun {Function?} Callback function for streaming.
+ * @return {String} The parsed template.
+ */
+ toHtml: null,
+
+
+ /**
+ * Helper method which provides you with a direct access to templates
+ * stored as HTML in the DOM. The DOM node with the given ID will be reated
+ * as a template, parsed and a new DOM node will be returned containing the
+ * parsed data.
+ *
+ * @param id {String} The id of the HTML template in the DOM.
+ * @param view {Object} The object holding the data to render.
+ * @param partials {Object} Object holding parts of a template.
+ * @return {DomNode} A DOM element holding the parsed template data.
+ */
+ get : function(id, view, partials) {
+ var template = document.getElementById(id);
+ var inner = template.innerHTML;
+
+ inner = this.toHtml(inner, view, partials);
+
+ var helper = qx.bom.Element.create("div");
+ helper.innerHTML = inner;
+
+ return helper.children[0];
+ }
+ }
+});
+
+(function() {
+
+/**
+ * Below is the original mustache.js code. Snapshot date is mentioned in
+ * the head of this file.
+ */
+
View
3  wrappers/requirejs/requirejs.mustache.js.tpl.post
@@ -0,0 +1,3 @@
+
+return Mustache;
+});
View
6 wrappers/requirejs/requirejs.mustache.js.tpl.pre
@@ -0,0 +1,6 @@
+/*
+Shameless port of a shameless port ^ 2
+@defunkt => @janl => @aq => @voodootikigod => @timruffles
+
+*/
+define(function(){
View
4 wrappers/yui3/mustache.js.tpl.post
@@ -0,0 +1,4 @@
+
+ Y.mustache = Mustache.to_html;
+
+}, "0");
View
1  wrappers/yui3/mustache.js.tpl.pre
@@ -0,0 +1 @@
+YUI.add("mustache", function(Y) {
Please sign in to comment.
Something went wrong with that request. Please try again.