From 38fb3d0b34cc423a2140b9863f7c9acfcdda1c0a Mon Sep 17 00:00:00 2001 From: addyosmani Date: Tue, 7 Aug 2012 16:54:55 +0100 Subject: [PATCH] initial commit. --- LICENSE | 23 + README.md | 4 +- Rakefile | 62 +++ TESTING.md | 75 +++ mustache.js | 436 ++++++++++++++++++ mustache.min.js | 6 + spec/_files/ampersand_escape.js | 3 + spec/_files/ampersand_escape.mustache | 1 + spec/_files/ampersand_escape.txt | 1 + spec/_files/apostrophe.js | 1 + spec/_files/apostrophe.mustache | 1 + spec/_files/apostrophe.txt | 1 + ...ay_of_partials_implicit_partial.2.mustache | 1 + .../array_of_partials_implicit_partial.js | 3 + ...rray_of_partials_implicit_partial.mustache | 4 + .../array_of_partials_implicit_partial.txt | 5 + .../array_of_partials_partial.2.mustache | 1 + spec/_files/array_of_partials_partial.js | 3 + .../_files/array_of_partials_partial.mustache | 4 + spec/_files/array_of_partials_partial.txt | 5 + spec/_files/array_of_strings.js | 1 + spec/_files/array_of_strings.mustache | 1 + spec/_files/array_of_strings.txt | 1 + spec/_files/array_of_strings_options.js | 1 + spec/_files/array_of_strings_options.mustache | 2 + spec/_files/array_of_strings_options.txt | 1 + spec/_files/array_partial.2.mustache | 4 + spec/_files/array_partial.js | 5 + spec/_files/array_partial.mustache | 1 + spec/_files/array_partial.txt | 6 + spec/_files/bug_11_eating_whitespace.js | 3 + spec/_files/bug_11_eating_whitespace.mustache | 1 + spec/_files/bug_11_eating_whitespace.txt | 1 + spec/_files/comments.js | 5 + spec/_files/comments.mustache | 1 + spec/_files/comments.txt | 1 + spec/_files/complex.js | 19 + spec/_files/complex.mustache | 16 + spec/_files/complex.txt | 6 + spec/_files/delimiters.js | 6 + spec/_files/delimiters.mustache | 7 + spec/_files/delimiters.txt | 5 + spec/_files/dot_notation.js | 23 + spec/_files/dot_notation.mustache | 9 + spec/_files/dot_notation.txt | 9 + spec/_files/double_render.js | 5 + spec/_files/double_render.mustache | 1 + spec/_files/double_render.txt | 1 + spec/_files/empty_partial.2.mustache | 1 + spec/_files/empty_partial.js | 3 + spec/_files/empty_partial.mustache | 2 + spec/_files/empty_partial.txt | 2 + spec/_files/empty_sections.js | 1 + spec/_files/empty_sections.mustache | 1 + spec/_files/empty_sections.txt | 1 + spec/_files/empty_template.js | 1 + spec/_files/empty_template.mustache | 1 + spec/_files/empty_template.txt | 1 + spec/_files/error_not_found.js | 1 + spec/_files/error_not_found.mustache | 1 + spec/_files/error_not_found.txt | 1 + spec/_files/escaped.js | 7 + spec/_files/escaped.mustache | 3 + spec/_files/escaped.txt | 3 + spec/_files/higher_order_sections.js | 9 + spec/_files/higher_order_sections.mustache | 1 + spec/_files/higher_order_sections.txt | 1 + spec/_files/inverted_section.js | 3 + spec/_files/inverted_section.mustache | 2 + spec/_files/inverted_section.txt | 1 + spec/_files/keys_with_questionmarks.js | 5 + spec/_files/keys_with_questionmarks.mustache | 3 + spec/_files/keys_with_questionmarks.txt | 1 + spec/_files/nesting.js | 7 + spec/_files/nesting.mustache | 5 + spec/_files/nesting.txt | 4 + spec/_files/null_string.js | 9 + spec/_files/null_string.mustache | 5 + spec/_files/null_string.txt | 5 + spec/_files/partial_recursion.2.mustache | 4 + spec/_files/partial_recursion.js | 11 + spec/_files/partial_recursion.mustache | 4 + spec/_files/partial_recursion.txt | 3 + spec/_files/recursion_with_same_names.js | 8 + .../_files/recursion_with_same_names.mustache | 7 + spec/_files/recursion_with_same_names.txt | 6 + spec/_files/reuse_of_enumerables.js | 6 + spec/_files/reuse_of_enumerables.mustache | 8 + spec/_files/reuse_of_enumerables.txt | 8 + spec/_files/section_as_context.js | 7 + spec/_files/section_as_context.mustache | 9 + spec/_files/section_as_context.txt | 6 + spec/_files/simple.js | 8 + spec/_files/simple.mustache | 5 + spec/_files/simple.txt | 3 + spec/_files/template_partial.2.mustache | 1 + spec/_files/template_partial.js | 8 + spec/_files/template_partial.mustache | 2 + spec/_files/template_partial.txt | 2 + spec/_files/two_in_a_row.js | 4 + spec/_files/two_in_a_row.mustache | 1 + spec/_files/two_in_a_row.txt | 1 + spec/_files/two_sections.js | 1 + spec/_files/two_sections.mustache | 4 + spec/_files/two_sections.txt | 1 + spec/_files/unescaped.js | 5 + spec/_files/unescaped.mustache | 1 + spec/_files/unescaped.txt | 1 + spec/_files/unknown_pragma.js | 1 + spec/_files/unknown_pragma.mustache | 1 + spec/_files/unknown_pragma.txt | 1 + spec/_files/view_partial.2.mustache | 5 + spec/_files/view_partial.js | 19 + spec/_files/view_partial.mustache | 3 + spec/_files/view_partial.txt | 6 + spec/_files/whitespace_partial.2.mustache | 5 + spec/_files/whitespace_partial.js | 19 + spec/_files/whitespace_partial.mustache | 3 + spec/_files/whitespace_partial.txt | 6 + spec/mustache_spec.rb | 276 +++++++++++ wrappers/commonjs/mustache.js.tpl.post | 8 + wrappers/commonjs/mustache.js.tpl.pre | 6 + wrappers/commonjs/package.json | 8 + wrappers/dojo/mustache.js.tpl.post | 4 + wrappers/dojo/mustache.js.tpl.pre | 9 + wrappers/jquery/jquery.mustache.js.tpl.post | 6 + wrappers/jquery/jquery.mustache.js.tpl.pre | 9 + wrappers/qooxdoo/qooxdoo.mustache.js.tpl.post | 9 + wrappers/qooxdoo/qooxdoo.mustache.js.tpl.pre | 127 +++++ .../requirejs/requirejs.mustache.js.tpl.post | 3 + .../requirejs/requirejs.mustache.js.tpl.pre | 6 + wrappers/yui3/mustache.js.tpl.post | 4 + wrappers/yui3/mustache.js.tpl.pre | 1 + 133 files changed, 1551 insertions(+), 2 deletions(-) create mode 100644 LICENSE create mode 100644 Rakefile create mode 100644 TESTING.md create mode 100644 mustache.js create mode 100644 mustache.min.js create mode 100644 spec/_files/ampersand_escape.js create mode 100644 spec/_files/ampersand_escape.mustache create mode 100644 spec/_files/ampersand_escape.txt create mode 100644 spec/_files/apostrophe.js create mode 100644 spec/_files/apostrophe.mustache create mode 100644 spec/_files/apostrophe.txt create mode 100644 spec/_files/array_of_partials_implicit_partial.2.mustache create mode 100644 spec/_files/array_of_partials_implicit_partial.js create mode 100644 spec/_files/array_of_partials_implicit_partial.mustache create mode 100644 spec/_files/array_of_partials_implicit_partial.txt create mode 100644 spec/_files/array_of_partials_partial.2.mustache create mode 100644 spec/_files/array_of_partials_partial.js create mode 100644 spec/_files/array_of_partials_partial.mustache create mode 100644 spec/_files/array_of_partials_partial.txt create mode 100644 spec/_files/array_of_strings.js create mode 100644 spec/_files/array_of_strings.mustache create mode 100644 spec/_files/array_of_strings.txt create mode 100644 spec/_files/array_of_strings_options.js create mode 100644 spec/_files/array_of_strings_options.mustache create mode 100644 spec/_files/array_of_strings_options.txt create mode 100644 spec/_files/array_partial.2.mustache create mode 100644 spec/_files/array_partial.js create mode 100644 spec/_files/array_partial.mustache create mode 100644 spec/_files/array_partial.txt create mode 100644 spec/_files/bug_11_eating_whitespace.js create mode 100644 spec/_files/bug_11_eating_whitespace.mustache create mode 100644 spec/_files/bug_11_eating_whitespace.txt create mode 100644 spec/_files/comments.js create mode 100644 spec/_files/comments.mustache create mode 100644 spec/_files/comments.txt create mode 100644 spec/_files/complex.js create mode 100644 spec/_files/complex.mustache create mode 100644 spec/_files/complex.txt create mode 100644 spec/_files/delimiters.js create mode 100644 spec/_files/delimiters.mustache create mode 100644 spec/_files/delimiters.txt create mode 100644 spec/_files/dot_notation.js create mode 100644 spec/_files/dot_notation.mustache create mode 100644 spec/_files/dot_notation.txt create mode 100644 spec/_files/double_render.js create mode 100644 spec/_files/double_render.mustache create mode 100644 spec/_files/double_render.txt create mode 100644 spec/_files/empty_partial.2.mustache create mode 100644 spec/_files/empty_partial.js create mode 100644 spec/_files/empty_partial.mustache create mode 100644 spec/_files/empty_partial.txt create mode 100644 spec/_files/empty_sections.js create mode 100644 spec/_files/empty_sections.mustache create mode 100644 spec/_files/empty_sections.txt create mode 100644 spec/_files/empty_template.js create mode 100644 spec/_files/empty_template.mustache create mode 100644 spec/_files/empty_template.txt create mode 100644 spec/_files/error_not_found.js create mode 100644 spec/_files/error_not_found.mustache create mode 100644 spec/_files/error_not_found.txt create mode 100644 spec/_files/escaped.js create mode 100644 spec/_files/escaped.mustache create mode 100644 spec/_files/escaped.txt create mode 100644 spec/_files/higher_order_sections.js create mode 100644 spec/_files/higher_order_sections.mustache create mode 100644 spec/_files/higher_order_sections.txt create mode 100644 spec/_files/inverted_section.js create mode 100644 spec/_files/inverted_section.mustache create mode 100644 spec/_files/inverted_section.txt create mode 100644 spec/_files/keys_with_questionmarks.js create mode 100644 spec/_files/keys_with_questionmarks.mustache create mode 100644 spec/_files/keys_with_questionmarks.txt create mode 100644 spec/_files/nesting.js create mode 100644 spec/_files/nesting.mustache create mode 100644 spec/_files/nesting.txt create mode 100644 spec/_files/null_string.js create mode 100644 spec/_files/null_string.mustache create mode 100644 spec/_files/null_string.txt create mode 100644 spec/_files/partial_recursion.2.mustache create mode 100644 spec/_files/partial_recursion.js create mode 100644 spec/_files/partial_recursion.mustache create mode 100644 spec/_files/partial_recursion.txt create mode 100644 spec/_files/recursion_with_same_names.js create mode 100644 spec/_files/recursion_with_same_names.mustache create mode 100644 spec/_files/recursion_with_same_names.txt create mode 100644 spec/_files/reuse_of_enumerables.js create mode 100644 spec/_files/reuse_of_enumerables.mustache create mode 100644 spec/_files/reuse_of_enumerables.txt create mode 100644 spec/_files/section_as_context.js create mode 100644 spec/_files/section_as_context.mustache create mode 100644 spec/_files/section_as_context.txt create mode 100644 spec/_files/simple.js create mode 100644 spec/_files/simple.mustache create mode 100644 spec/_files/simple.txt create mode 100644 spec/_files/template_partial.2.mustache create mode 100644 spec/_files/template_partial.js create mode 100644 spec/_files/template_partial.mustache create mode 100644 spec/_files/template_partial.txt create mode 100644 spec/_files/two_in_a_row.js create mode 100644 spec/_files/two_in_a_row.mustache create mode 100644 spec/_files/two_in_a_row.txt create mode 100644 spec/_files/two_sections.js create mode 100644 spec/_files/two_sections.mustache create mode 100644 spec/_files/two_sections.txt create mode 100644 spec/_files/unescaped.js create mode 100644 spec/_files/unescaped.mustache create mode 100644 spec/_files/unescaped.txt create mode 100644 spec/_files/unknown_pragma.js create mode 100644 spec/_files/unknown_pragma.mustache create mode 100644 spec/_files/unknown_pragma.txt create mode 100644 spec/_files/view_partial.2.mustache create mode 100644 spec/_files/view_partial.js create mode 100644 spec/_files/view_partial.mustache create mode 100644 spec/_files/view_partial.txt create mode 100644 spec/_files/whitespace_partial.2.mustache create mode 100644 spec/_files/whitespace_partial.js create mode 100644 spec/_files/whitespace_partial.mustache create mode 100644 spec/_files/whitespace_partial.txt create mode 100644 spec/mustache_spec.rb create mode 100644 wrappers/commonjs/mustache.js.tpl.post create mode 100644 wrappers/commonjs/mustache.js.tpl.pre create mode 100644 wrappers/commonjs/package.json create mode 100644 wrappers/dojo/mustache.js.tpl.post create mode 100644 wrappers/dojo/mustache.js.tpl.pre create mode 100644 wrappers/jquery/jquery.mustache.js.tpl.post create mode 100644 wrappers/jquery/jquery.mustache.js.tpl.pre create mode 100644 wrappers/qooxdoo/qooxdoo.mustache.js.tpl.post create mode 100644 wrappers/qooxdoo/qooxdoo.mustache.js.tpl.pre create mode 100644 wrappers/requirejs/requirejs.mustache.js.tpl.post create mode 100644 wrappers/requirejs/requirejs.mustache.js.tpl.pre create mode 100644 wrappers/yui3/mustache.js.tpl.post create mode 100644 wrappers/yui3/mustache.js.tpl.pre diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..395e263 --- /dev/null +++ b/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. \ No newline at end of file diff --git a/README.md b/README.md index 9a7cc94..3f6b123 100644 --- a/README.md +++ b/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. diff --git a/Rakefile b/Rakefile new file mode 100644 index 0000000..f546b74 --- /dev/null +++ b/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 diff --git a/TESTING.md b/TESTING.md new file mode 100644 index 0000000..0be439b --- /dev/null +++ b/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` diff --git a/mustache.js b/mustache.js new file mode 100644 index 0000000..4ac3fae --- /dev/null +++ b/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 = { + "&": "&", + "<": "<", + ">": ">", + '"': '"', + "'": ''' + }; + + 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"); + } + } + }); +}(); diff --git a/mustache.min.js b/mustache.min.js new file mode 100644 index 0000000..8014bcf --- /dev/null +++ b/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={"&":"&","<":"<",">":">",'"':""","'":"'"},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|&|\\{|%)?([^#\\^]+?)\\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;k0)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" +}; diff --git a/spec/_files/ampersand_escape.mustache b/spec/_files/ampersand_escape.mustache new file mode 100644 index 0000000..6501a48 --- /dev/null +++ b/spec/_files/ampersand_escape.mustache @@ -0,0 +1 @@ +{{&message}} diff --git a/spec/_files/ampersand_escape.txt b/spec/_files/ampersand_escape.txt new file mode 100644 index 0000000..2ed3fd3 --- /dev/null +++ b/spec/_files/ampersand_escape.txt @@ -0,0 +1 @@ +Some diff --git a/spec/_files/apostrophe.js b/spec/_files/apostrophe.js new file mode 100644 index 0000000..df69cd2 --- /dev/null +++ b/spec/_files/apostrophe.js @@ -0,0 +1 @@ +var apostrophe = {'apos': "'", 'control':'X'}; diff --git a/spec/_files/apostrophe.mustache b/spec/_files/apostrophe.mustache new file mode 100644 index 0000000..e8687aa --- /dev/null +++ b/spec/_files/apostrophe.mustache @@ -0,0 +1 @@ +{{apos}}{{control}} diff --git a/spec/_files/apostrophe.txt b/spec/_files/apostrophe.txt new file mode 100644 index 0000000..4427c30 --- /dev/null +++ b/spec/_files/apostrophe.txt @@ -0,0 +1 @@ +'X diff --git a/spec/_files/array_of_partials_implicit_partial.2.mustache b/spec/_files/array_of_partials_implicit_partial.2.mustache new file mode 100644 index 0000000..12f7159 --- /dev/null +++ b/spec/_files/array_of_partials_implicit_partial.2.mustache @@ -0,0 +1 @@ +{{.}} diff --git a/spec/_files/array_of_partials_implicit_partial.js b/spec/_files/array_of_partials_implicit_partial.js new file mode 100644 index 0000000..fcdf3b0 --- /dev/null +++ b/spec/_files/array_of_partials_implicit_partial.js @@ -0,0 +1,3 @@ +var partial_context = { + numbers: ['1', '2', '3', '4'] +}; diff --git a/spec/_files/array_of_partials_implicit_partial.mustache b/spec/_files/array_of_partials_implicit_partial.mustache new file mode 100644 index 0000000..1af6d68 --- /dev/null +++ b/spec/_files/array_of_partials_implicit_partial.mustache @@ -0,0 +1,4 @@ +Here is some stuff! +{{#numbers}} +{{>partial}} +{{/numbers}} diff --git a/spec/_files/array_of_partials_implicit_partial.txt b/spec/_files/array_of_partials_implicit_partial.txt new file mode 100644 index 0000000..f622375 --- /dev/null +++ b/spec/_files/array_of_partials_implicit_partial.txt @@ -0,0 +1,5 @@ +Here is some stuff! +1 +2 +3 +4 diff --git a/spec/_files/array_of_partials_partial.2.mustache b/spec/_files/array_of_partials_partial.2.mustache new file mode 100644 index 0000000..bdde77d --- /dev/null +++ b/spec/_files/array_of_partials_partial.2.mustache @@ -0,0 +1 @@ +{{i}} diff --git a/spec/_files/array_of_partials_partial.js b/spec/_files/array_of_partials_partial.js new file mode 100644 index 0000000..45611cf --- /dev/null +++ b/spec/_files/array_of_partials_partial.js @@ -0,0 +1,3 @@ +var partial_context = { + numbers: [{i: '1'}, {i: '2'}, {i: '3'}, {i: '4'}] +}; diff --git a/spec/_files/array_of_partials_partial.mustache b/spec/_files/array_of_partials_partial.mustache new file mode 100644 index 0000000..1af6d68 --- /dev/null +++ b/spec/_files/array_of_partials_partial.mustache @@ -0,0 +1,4 @@ +Here is some stuff! +{{#numbers}} +{{>partial}} +{{/numbers}} diff --git a/spec/_files/array_of_partials_partial.txt b/spec/_files/array_of_partials_partial.txt new file mode 100644 index 0000000..f622375 --- /dev/null +++ b/spec/_files/array_of_partials_partial.txt @@ -0,0 +1,5 @@ +Here is some stuff! +1 +2 +3 +4 diff --git a/spec/_files/array_of_strings.js b/spec/_files/array_of_strings.js new file mode 100644 index 0000000..12c4992 --- /dev/null +++ b/spec/_files/array_of_strings.js @@ -0,0 +1 @@ +var array_of_strings = {array_of_strings: ['hello', 'world']}; diff --git a/spec/_files/array_of_strings.mustache b/spec/_files/array_of_strings.mustache new file mode 100644 index 0000000..2898705 --- /dev/null +++ b/spec/_files/array_of_strings.mustache @@ -0,0 +1 @@ +{{#array_of_strings}}{{.}} {{/array_of_strings}} \ No newline at end of file diff --git a/spec/_files/array_of_strings.txt b/spec/_files/array_of_strings.txt new file mode 100644 index 0000000..4a1f475 --- /dev/null +++ b/spec/_files/array_of_strings.txt @@ -0,0 +1 @@ +hello world diff --git a/spec/_files/array_of_strings_options.js b/spec/_files/array_of_strings_options.js new file mode 100644 index 0000000..2e29adf --- /dev/null +++ b/spec/_files/array_of_strings_options.js @@ -0,0 +1 @@ +var array_of_strings_options = {array_of_strings_options: ['hello', 'world']}; diff --git a/spec/_files/array_of_strings_options.mustache b/spec/_files/array_of_strings_options.mustache new file mode 100644 index 0000000..6b28278 --- /dev/null +++ b/spec/_files/array_of_strings_options.mustache @@ -0,0 +1,2 @@ +{{%IMPLICIT-ITERATOR iterator=rob}} +{{#array_of_strings_options}}{{rob}} {{/array_of_strings_options}} \ No newline at end of file diff --git a/spec/_files/array_of_strings_options.txt b/spec/_files/array_of_strings_options.txt new file mode 100644 index 0000000..4a1f475 --- /dev/null +++ b/spec/_files/array_of_strings_options.txt @@ -0,0 +1 @@ +hello world diff --git a/spec/_files/array_partial.2.mustache b/spec/_files/array_partial.2.mustache new file mode 100644 index 0000000..235dd5f --- /dev/null +++ b/spec/_files/array_partial.2.mustache @@ -0,0 +1,4 @@ +Here's a non-sense array of values +{{#array}} + {{.}} +{{/array}} \ No newline at end of file diff --git a/spec/_files/array_partial.js b/spec/_files/array_partial.js new file mode 100644 index 0000000..88d7592 --- /dev/null +++ b/spec/_files/array_partial.js @@ -0,0 +1,5 @@ +var partial_context = { + partial: { + array: ['1', '2', '3', '4'] + } +}; \ No newline at end of file diff --git a/spec/_files/array_partial.mustache b/spec/_files/array_partial.mustache new file mode 100644 index 0000000..7a336fe --- /dev/null +++ b/spec/_files/array_partial.mustache @@ -0,0 +1 @@ +{{>partial}} \ No newline at end of file diff --git a/spec/_files/array_partial.txt b/spec/_files/array_partial.txt new file mode 100644 index 0000000..fd5f48e --- /dev/null +++ b/spec/_files/array_partial.txt @@ -0,0 +1,6 @@ +Here's a non-sense array of values + 1 + 2 + 3 + 4 + diff --git a/spec/_files/bug_11_eating_whitespace.js b/spec/_files/bug_11_eating_whitespace.js new file mode 100644 index 0000000..78ce975 --- /dev/null +++ b/spec/_files/bug_11_eating_whitespace.js @@ -0,0 +1,3 @@ +var bug_11_eating_whitespace = { + tag: "yo" +}; diff --git a/spec/_files/bug_11_eating_whitespace.mustache b/spec/_files/bug_11_eating_whitespace.mustache new file mode 100644 index 0000000..8d5cd92 --- /dev/null +++ b/spec/_files/bug_11_eating_whitespace.mustache @@ -0,0 +1 @@ +{{tag}} foo diff --git a/spec/_files/bug_11_eating_whitespace.txt b/spec/_files/bug_11_eating_whitespace.txt new file mode 100644 index 0000000..f5bbc85 --- /dev/null +++ b/spec/_files/bug_11_eating_whitespace.txt @@ -0,0 +1 @@ +yo foo diff --git a/spec/_files/comments.js b/spec/_files/comments.js new file mode 100644 index 0000000..f85f979 --- /dev/null +++ b/spec/_files/comments.js @@ -0,0 +1,5 @@ +var comments = { + title: function() { + return "A Comedy of Errors"; + } +}; diff --git a/spec/_files/comments.mustache b/spec/_files/comments.mustache new file mode 100644 index 0000000..5036801 --- /dev/null +++ b/spec/_files/comments.mustache @@ -0,0 +1 @@ +

{{title}}{{! just something interesting... or not... }}

diff --git a/spec/_files/comments.txt b/spec/_files/comments.txt new file mode 100644 index 0000000..0133517 --- /dev/null +++ b/spec/_files/comments.txt @@ -0,0 +1 @@ +

A Comedy of Errors

diff --git a/spec/_files/complex.js b/spec/_files/complex.js new file mode 100644 index 0000000..a0dfb63 --- /dev/null +++ b/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; + } +}; diff --git a/spec/_files/complex.mustache b/spec/_files/complex.mustache new file mode 100644 index 0000000..23bec3c --- /dev/null +++ b/spec/_files/complex.mustache @@ -0,0 +1,16 @@ +

{{header}}

+{{#list}} +
    + {{#item}} + {{#current}} +
  • {{name}}
  • + {{/current}} + {{#link}} +
  • {{name}}
  • + {{/link}} + {{/item}} +
+{{/list}} +{{#empty}} +

The list is empty.

+{{/empty}} \ No newline at end of file diff --git a/spec/_files/complex.txt b/spec/_files/complex.txt new file mode 100644 index 0000000..b51ca10 --- /dev/null +++ b/spec/_files/complex.txt @@ -0,0 +1,6 @@ +

Colors

+ diff --git a/spec/_files/delimiters.js b/spec/_files/delimiters.js new file mode 100644 index 0000000..220d0d2 --- /dev/null +++ b/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!." +} diff --git a/spec/_files/delimiters.mustache b/spec/_files/delimiters.mustache new file mode 100644 index 0000000..a73e061 --- /dev/null +++ b/spec/_files/delimiters.mustache @@ -0,0 +1,7 @@ +{{=<% %>=}}* +<% first %> +* <% second %> +<%=| |=%> +* | third | +|={{ }}=| +* {{ fourth }} \ No newline at end of file diff --git a/spec/_files/delimiters.txt b/spec/_files/delimiters.txt new file mode 100644 index 0000000..698a6bb --- /dev/null +++ b/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!. diff --git a/spec/_files/dot_notation.js b/spec/_files/dot_notation.js new file mode 100644 index 0000000..c1295f5 --- /dev/null +++ b/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: '€', + name: 'Euro' + } + }, + availability:{ + status: true, + text: "In Stock" + }, + // And now, some truthy false values + truthy: { + zero: 0, + notTrue: false + } +}; diff --git a/spec/_files/dot_notation.mustache b/spec/_files/dot_notation.mustache new file mode 100644 index 0000000..da1bad7 --- /dev/null +++ b/spec/_files/dot_notation.mustache @@ -0,0 +1,9 @@ + +

{{name}}

+

Authors:

    {{#authors}}
  • {{.}}
  • {{/authors}}

+

Price: {{price.currency.symbol}}{{price.value}} {{#price.currency}}{{name}} {{availability.text}}{{/price.currency}}

+

VAT: {{price.currency.symbol}}{{price.vat}}

+ +

Test truthy false values:

+

Zero: {{truthy.zero}}

+

False: {{truthy.notTrue}}

diff --git a/spec/_files/dot_notation.txt b/spec/_files/dot_notation.txt new file mode 100644 index 0000000..d0e4707 --- /dev/null +++ b/spec/_files/dot_notation.txt @@ -0,0 +1,9 @@ + +

A Book

+

Authors:

  • John Power
  • Jamie Walsh

+

Price: €200 Euro In Stock

+

VAT: €40

+ +

Test truthy false values:

+

Zero: 0

+

False: false

diff --git a/spec/_files/double_render.js b/spec/_files/double_render.js new file mode 100644 index 0000000..24125dc --- /dev/null +++ b/spec/_files/double_render.js @@ -0,0 +1,5 @@ +var double_render = { + foo: true, + bar: "{{win}}", + win: "FAIL" +}; \ No newline at end of file diff --git a/spec/_files/double_render.mustache b/spec/_files/double_render.mustache new file mode 100644 index 0000000..4500fd7 --- /dev/null +++ b/spec/_files/double_render.mustache @@ -0,0 +1 @@ +{{#foo}}{{bar}}{{/foo}} diff --git a/spec/_files/double_render.txt b/spec/_files/double_render.txt new file mode 100644 index 0000000..b6e652d --- /dev/null +++ b/spec/_files/double_render.txt @@ -0,0 +1 @@ +{{win}} diff --git a/spec/_files/empty_partial.2.mustache b/spec/_files/empty_partial.2.mustache new file mode 100644 index 0000000..b920295 --- /dev/null +++ b/spec/_files/empty_partial.2.mustache @@ -0,0 +1 @@ +yo \ No newline at end of file diff --git a/spec/_files/empty_partial.js b/spec/_files/empty_partial.js new file mode 100644 index 0000000..9cc53c6 --- /dev/null +++ b/spec/_files/empty_partial.js @@ -0,0 +1,3 @@ +var partial_context = { + foo: 1 +}; diff --git a/spec/_files/empty_partial.mustache b/spec/_files/empty_partial.mustache new file mode 100644 index 0000000..a710047 --- /dev/null +++ b/spec/_files/empty_partial.mustache @@ -0,0 +1,2 @@ +hey {{foo}} +{{>partial}} diff --git a/spec/_files/empty_partial.txt b/spec/_files/empty_partial.txt new file mode 100644 index 0000000..90d2b9f --- /dev/null +++ b/spec/_files/empty_partial.txt @@ -0,0 +1,2 @@ +hey 1 +yo diff --git a/spec/_files/empty_sections.js b/spec/_files/empty_sections.js new file mode 100644 index 0000000..6e50514 --- /dev/null +++ b/spec/_files/empty_sections.js @@ -0,0 +1 @@ +var empty_sections = {}; diff --git a/spec/_files/empty_sections.mustache b/spec/_files/empty_sections.mustache new file mode 100644 index 0000000..b6065db --- /dev/null +++ b/spec/_files/empty_sections.mustache @@ -0,0 +1 @@ +{{#foo}}{{/foo}}foo{{#bar}}{{/bar}} diff --git a/spec/_files/empty_sections.txt b/spec/_files/empty_sections.txt new file mode 100644 index 0000000..257cc56 --- /dev/null +++ b/spec/_files/empty_sections.txt @@ -0,0 +1 @@ +foo diff --git a/spec/_files/empty_template.js b/spec/_files/empty_template.js new file mode 100644 index 0000000..564c9e2 --- /dev/null +++ b/spec/_files/empty_template.js @@ -0,0 +1 @@ +var empty_template = {}; diff --git a/spec/_files/empty_template.mustache b/spec/_files/empty_template.mustache new file mode 100644 index 0000000..bb2367a --- /dev/null +++ b/spec/_files/empty_template.mustache @@ -0,0 +1 @@ +

Test

\ No newline at end of file diff --git a/spec/_files/empty_template.txt b/spec/_files/empty_template.txt new file mode 100644 index 0000000..60652b6 --- /dev/null +++ b/spec/_files/empty_template.txt @@ -0,0 +1 @@ +

Test

diff --git a/spec/_files/error_not_found.js b/spec/_files/error_not_found.js new file mode 100644 index 0000000..6cdbdb3 --- /dev/null +++ b/spec/_files/error_not_found.js @@ -0,0 +1 @@ +var error_not_found = {bar: 2}; \ No newline at end of file diff --git a/spec/_files/error_not_found.mustache b/spec/_files/error_not_found.mustache new file mode 100644 index 0000000..24369f7 --- /dev/null +++ b/spec/_files/error_not_found.mustache @@ -0,0 +1 @@ +{{foo}} \ No newline at end of file diff --git a/spec/_files/error_not_found.txt b/spec/_files/error_not_found.txt new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/spec/_files/error_not_found.txt @@ -0,0 +1 @@ + diff --git a/spec/_files/escaped.js b/spec/_files/escaped.js new file mode 100644 index 0000000..875776a --- /dev/null +++ b/spec/_files/escaped.js @@ -0,0 +1,7 @@ +var escaped = { + title: function() { + return "Bear > Shark"; + }, + subtitle: 'Rock 'n Roll', + entities: """ +}; diff --git a/spec/_files/escaped.mustache b/spec/_files/escaped.mustache new file mode 100644 index 0000000..b8f82d0 --- /dev/null +++ b/spec/_files/escaped.mustache @@ -0,0 +1,3 @@ +

{{title}}

+

{{subtitle}}

+But not {{entities}}. diff --git a/spec/_files/escaped.txt b/spec/_files/escaped.txt new file mode 100644 index 0000000..7f750d4 --- /dev/null +++ b/spec/_files/escaped.txt @@ -0,0 +1,3 @@ +

Bear > Shark

+

Rock 'n Roll

+But not ". diff --git a/spec/_files/higher_order_sections.js b/spec/_files/higher_order_sections.js new file mode 100644 index 0000000..c7e558e --- /dev/null +++ b/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 "" + render(text) + ' ' + this.helper; + } + } +} \ No newline at end of file diff --git a/spec/_files/higher_order_sections.mustache b/spec/_files/higher_order_sections.mustache new file mode 100644 index 0000000..04f5318 --- /dev/null +++ b/spec/_files/higher_order_sections.mustache @@ -0,0 +1 @@ +{{#bolder}}Hi {{name}}.{{/bolder}} diff --git a/spec/_files/higher_order_sections.txt b/spec/_files/higher_order_sections.txt new file mode 100644 index 0000000..9db786a --- /dev/null +++ b/spec/_files/higher_order_sections.txt @@ -0,0 +1 @@ +Hi Tater. To tinker? diff --git a/spec/_files/inverted_section.js b/spec/_files/inverted_section.js new file mode 100644 index 0000000..cb96ecf --- /dev/null +++ b/spec/_files/inverted_section.js @@ -0,0 +1,3 @@ +var inverted_section = { + "repo": [] +} diff --git a/spec/_files/inverted_section.mustache b/spec/_files/inverted_section.mustache new file mode 100644 index 0000000..beec558 --- /dev/null +++ b/spec/_files/inverted_section.mustache @@ -0,0 +1,2 @@ +{{#repo}}{{name}}{{/repo}} +{{^repo}}No repos :({{/repo}} diff --git a/spec/_files/inverted_section.txt b/spec/_files/inverted_section.txt new file mode 100644 index 0000000..5fd0de1 --- /dev/null +++ b/spec/_files/inverted_section.txt @@ -0,0 +1 @@ +No repos :( diff --git a/spec/_files/keys_with_questionmarks.js b/spec/_files/keys_with_questionmarks.js new file mode 100644 index 0000000..55a220d --- /dev/null +++ b/spec/_files/keys_with_questionmarks.js @@ -0,0 +1,5 @@ +var keys_with_questionmarks = { + "person?": { + name: "Jon" + } +} diff --git a/spec/_files/keys_with_questionmarks.mustache b/spec/_files/keys_with_questionmarks.mustache new file mode 100644 index 0000000..417f17f --- /dev/null +++ b/spec/_files/keys_with_questionmarks.mustache @@ -0,0 +1,3 @@ +{{#person?}} + Hi {{name}}! +{{/person?}} diff --git a/spec/_files/keys_with_questionmarks.txt b/spec/_files/keys_with_questionmarks.txt new file mode 100644 index 0000000..0f69b94 --- /dev/null +++ b/spec/_files/keys_with_questionmarks.txt @@ -0,0 +1 @@ + Hi Jon! diff --git a/spec/_files/nesting.js b/spec/_files/nesting.js new file mode 100644 index 0000000..1f76cd0 --- /dev/null +++ b/spec/_files/nesting.js @@ -0,0 +1,7 @@ +var nesting = { + foo: [ + {a: {b: 1}}, + {a: {b: 2}}, + {a: {b: 3}} + ] +}; diff --git a/spec/_files/nesting.mustache b/spec/_files/nesting.mustache new file mode 100644 index 0000000..551366d --- /dev/null +++ b/spec/_files/nesting.mustache @@ -0,0 +1,5 @@ +{{#foo}} + {{#a}} + {{b}} + {{/a}} +{{/foo}} diff --git a/spec/_files/nesting.txt b/spec/_files/nesting.txt new file mode 100644 index 0000000..caf5afd --- /dev/null +++ b/spec/_files/nesting.txt @@ -0,0 +1,4 @@ + 1 + 2 + 3 + diff --git a/spec/_files/null_string.js b/spec/_files/null_string.js new file mode 100644 index 0000000..93414a1 --- /dev/null +++ b/spec/_files/null_string.js @@ -0,0 +1,9 @@ +var null_string = { + name: "Elise", + glytch: true, + binary: false, + value: null, + numeric: function() { + return NaN; + } +}; diff --git a/spec/_files/null_string.mustache b/spec/_files/null_string.mustache new file mode 100644 index 0000000..12bcb08 --- /dev/null +++ b/spec/_files/null_string.mustache @@ -0,0 +1,5 @@ +Hello {{name}} +glytch {{glytch}} +binary {{binary}} +value {{value}} +numeric {{numeric}} \ No newline at end of file diff --git a/spec/_files/null_string.txt b/spec/_files/null_string.txt new file mode 100644 index 0000000..827569b --- /dev/null +++ b/spec/_files/null_string.txt @@ -0,0 +1,5 @@ +Hello Elise +glytch true +binary false +value +numeric NaN diff --git a/spec/_files/partial_recursion.2.mustache b/spec/_files/partial_recursion.2.mustache new file mode 100644 index 0000000..457d2a0 --- /dev/null +++ b/spec/_files/partial_recursion.2.mustache @@ -0,0 +1,4 @@ +{{name}} +{{#children}} +{{>partial}} +{{/children}} \ No newline at end of file diff --git a/spec/_files/partial_recursion.js b/spec/_files/partial_recursion.js new file mode 100644 index 0000000..ad1f2eb --- /dev/null +++ b/spec/_files/partial_recursion.js @@ -0,0 +1,11 @@ +var partial_context = { + name: '1', + kids: [ + { + name: '1.1', + children: [ + {name: '1.1.1'} + ] + } + ] +}; diff --git a/spec/_files/partial_recursion.mustache b/spec/_files/partial_recursion.mustache new file mode 100644 index 0000000..d965a6a --- /dev/null +++ b/spec/_files/partial_recursion.mustache @@ -0,0 +1,4 @@ +{{name}} +{{#kids}} +{{>partial}} +{{/kids}} \ No newline at end of file diff --git a/spec/_files/partial_recursion.txt b/spec/_files/partial_recursion.txt new file mode 100644 index 0000000..0f70515 --- /dev/null +++ b/spec/_files/partial_recursion.txt @@ -0,0 +1,3 @@ +1 +1.1 +1.1.1 diff --git a/spec/_files/recursion_with_same_names.js b/spec/_files/recursion_with_same_names.js new file mode 100644 index 0000000..5cceb08 --- /dev/null +++ b/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}, + ] +}; \ No newline at end of file diff --git a/spec/_files/recursion_with_same_names.mustache b/spec/_files/recursion_with_same_names.mustache new file mode 100644 index 0000000..c331d04 --- /dev/null +++ b/spec/_files/recursion_with_same_names.mustache @@ -0,0 +1,7 @@ +{{ name }} +{{ description }} + +{{#terms}} + {{name}} + {{index}} +{{/terms}} diff --git a/spec/_files/recursion_with_same_names.txt b/spec/_files/recursion_with_same_names.txt new file mode 100644 index 0000000..c23bb65 --- /dev/null +++ b/spec/_files/recursion_with_same_names.txt @@ -0,0 +1,6 @@ +name +desc + t1 + 0 + t2 + 1 diff --git a/spec/_files/reuse_of_enumerables.js b/spec/_files/reuse_of_enumerables.js new file mode 100644 index 0000000..543e121 --- /dev/null +++ b/spec/_files/reuse_of_enumerables.js @@ -0,0 +1,6 @@ +var reuse_of_enumerables = { + terms: [ + {name: 't1', index: 0}, + {name: 't2', index: 1}, + ] +}; \ No newline at end of file diff --git a/spec/_files/reuse_of_enumerables.mustache b/spec/_files/reuse_of_enumerables.mustache new file mode 100644 index 0000000..cc0cb7a --- /dev/null +++ b/spec/_files/reuse_of_enumerables.mustache @@ -0,0 +1,8 @@ +{{#terms}} + {{name}} + {{index}} +{{/terms}} +{{#terms}} + {{name}} + {{index}} +{{/terms}} diff --git a/spec/_files/reuse_of_enumerables.txt b/spec/_files/reuse_of_enumerables.txt new file mode 100644 index 0000000..6d05d96 --- /dev/null +++ b/spec/_files/reuse_of_enumerables.txt @@ -0,0 +1,8 @@ + t1 + 0 + t2 + 1 + t1 + 0 + t2 + 1 diff --git a/spec/_files/section_as_context.js b/spec/_files/section_as_context.js new file mode 100644 index 0000000..81ca1be --- /dev/null +++ b/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'}] + } +}; diff --git a/spec/_files/section_as_context.mustache b/spec/_files/section_as_context.mustache new file mode 100644 index 0000000..59990f6 --- /dev/null +++ b/spec/_files/section_as_context.mustache @@ -0,0 +1,9 @@ +{{#a_object}} +

{{title}}

+

{{description}}

+
    + {{#a_list}} +
  • {{label}}
  • + {{/a_list}} +
+{{/a_object}} diff --git a/spec/_files/section_as_context.txt b/spec/_files/section_as_context.txt new file mode 100644 index 0000000..53ee336 --- /dev/null +++ b/spec/_files/section_as_context.txt @@ -0,0 +1,6 @@ +

this is an object

+

one of its attributes is a list

+
    +
  • listitem1
  • +
  • listitem2
  • +
diff --git a/spec/_files/simple.js b/spec/_files/simple.js new file mode 100644 index 0000000..30f9834 --- /dev/null +++ b/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 +}; diff --git a/spec/_files/simple.mustache b/spec/_files/simple.mustache new file mode 100644 index 0000000..03df206 --- /dev/null +++ b/spec/_files/simple.mustache @@ -0,0 +1,5 @@ +Hello {{name}} +You have just won ${{value}}! +{{#in_ca}} +Well, ${{ taxed_value }}, after taxes. +{{/in_ca}} \ No newline at end of file diff --git a/spec/_files/simple.txt b/spec/_files/simple.txt new file mode 100644 index 0000000..5d75d65 --- /dev/null +++ b/spec/_files/simple.txt @@ -0,0 +1,3 @@ +Hello Chris +You have just won $10000! +Well, $6000, after taxes. diff --git a/spec/_files/template_partial.2.mustache b/spec/_files/template_partial.2.mustache new file mode 100644 index 0000000..54bdd7d --- /dev/null +++ b/spec/_files/template_partial.2.mustache @@ -0,0 +1 @@ +Again, {{again}}! \ No newline at end of file diff --git a/spec/_files/template_partial.js b/spec/_files/template_partial.js new file mode 100644 index 0000000..68a4317 --- /dev/null +++ b/spec/_files/template_partial.js @@ -0,0 +1,8 @@ +var partial_context = { + title: function() { + return "Welcome"; + }, + partial: { + again: "Goodbye" + } +} diff --git a/spec/_files/template_partial.mustache b/spec/_files/template_partial.mustache new file mode 100644 index 0000000..6a09372 --- /dev/null +++ b/spec/_files/template_partial.mustache @@ -0,0 +1,2 @@ +

{{title}}

+{{>partial}} \ No newline at end of file diff --git a/spec/_files/template_partial.txt b/spec/_files/template_partial.txt new file mode 100644 index 0000000..692698f --- /dev/null +++ b/spec/_files/template_partial.txt @@ -0,0 +1,2 @@ +

Welcome

+Again, Goodbye! diff --git a/spec/_files/two_in_a_row.js b/spec/_files/two_in_a_row.js new file mode 100644 index 0000000..09c1809 --- /dev/null +++ b/spec/_files/two_in_a_row.js @@ -0,0 +1,4 @@ +var two_in_a_row = { + name: "Joe", + greeting: "Welcome" +}; diff --git a/spec/_files/two_in_a_row.mustache b/spec/_files/two_in_a_row.mustache new file mode 100644 index 0000000..dc38602 --- /dev/null +++ b/spec/_files/two_in_a_row.mustache @@ -0,0 +1 @@ +{{greeting}}, {{name}}! \ No newline at end of file diff --git a/spec/_files/two_in_a_row.txt b/spec/_files/two_in_a_row.txt new file mode 100644 index 0000000..c6d6a9b --- /dev/null +++ b/spec/_files/two_in_a_row.txt @@ -0,0 +1 @@ +Welcome, Joe! diff --git a/spec/_files/two_sections.js b/spec/_files/two_sections.js new file mode 100644 index 0000000..8546f64 --- /dev/null +++ b/spec/_files/two_sections.js @@ -0,0 +1 @@ +var two_sections = {}; \ No newline at end of file diff --git a/spec/_files/two_sections.mustache b/spec/_files/two_sections.mustache new file mode 100644 index 0000000..a4b9f2a --- /dev/null +++ b/spec/_files/two_sections.mustache @@ -0,0 +1,4 @@ +{{#foo}} +{{/foo}} +{{#bar}} +{{/bar}} diff --git a/spec/_files/two_sections.txt b/spec/_files/two_sections.txt new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/spec/_files/two_sections.txt @@ -0,0 +1 @@ + diff --git a/spec/_files/unescaped.js b/spec/_files/unescaped.js new file mode 100644 index 0000000..0bd20b8 --- /dev/null +++ b/spec/_files/unescaped.js @@ -0,0 +1,5 @@ +var unescaped = { + title: function() { + return "Bear > Shark"; + } +}; diff --git a/spec/_files/unescaped.mustache b/spec/_files/unescaped.mustache new file mode 100644 index 0000000..9982708 --- /dev/null +++ b/spec/_files/unescaped.mustache @@ -0,0 +1 @@ +

{{{title}}}

\ No newline at end of file diff --git a/spec/_files/unescaped.txt b/spec/_files/unescaped.txt new file mode 100644 index 0000000..089ad79 --- /dev/null +++ b/spec/_files/unescaped.txt @@ -0,0 +1 @@ +

Bear > Shark

diff --git a/spec/_files/unknown_pragma.js b/spec/_files/unknown_pragma.js new file mode 100644 index 0000000..68344a5 --- /dev/null +++ b/spec/_files/unknown_pragma.js @@ -0,0 +1 @@ +var unknown_pragma = {}; diff --git a/spec/_files/unknown_pragma.mustache b/spec/_files/unknown_pragma.mustache new file mode 100644 index 0000000..113810d --- /dev/null +++ b/spec/_files/unknown_pragma.mustache @@ -0,0 +1 @@ +{{%I-HAVE-THE-GREATEST-MUSTACHE}} diff --git a/spec/_files/unknown_pragma.txt b/spec/_files/unknown_pragma.txt new file mode 100644 index 0000000..a34840d --- /dev/null +++ b/spec/_files/unknown_pragma.txt @@ -0,0 +1 @@ +ERROR: This implementation of mustache doesn't understand the 'I-HAVE-THE-GREATEST-MUSTACHE' pragma diff --git a/spec/_files/view_partial.2.mustache b/spec/_files/view_partial.2.mustache new file mode 100644 index 0000000..2fea632 --- /dev/null +++ b/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}} diff --git a/spec/_files/view_partial.js b/spec/_files/view_partial.js new file mode 100644 index 0000000..30ade55 --- /dev/null +++ b/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 + } +}; + diff --git a/spec/_files/view_partial.mustache b/spec/_files/view_partial.mustache new file mode 100644 index 0000000..8f0c08d --- /dev/null +++ b/spec/_files/view_partial.mustache @@ -0,0 +1,3 @@ +

{{greeting}}

+{{>partial}} +

{{farewell}}

\ No newline at end of file diff --git a/spec/_files/view_partial.txt b/spec/_files/view_partial.txt new file mode 100644 index 0000000..160b0b6 --- /dev/null +++ b/spec/_files/view_partial.txt @@ -0,0 +1,6 @@ +

Welcome

+Hello Chris +You have just won $10000! +Well, $6000, after taxes. + +

Fair enough, right?

diff --git a/spec/_files/whitespace_partial.2.mustache b/spec/_files/whitespace_partial.2.mustache new file mode 100644 index 0000000..9c46084 --- /dev/null +++ b/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 }} diff --git a/spec/_files/whitespace_partial.js b/spec/_files/whitespace_partial.js new file mode 100644 index 0000000..30ade55 --- /dev/null +++ b/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 + } +}; + diff --git a/spec/_files/whitespace_partial.mustache b/spec/_files/whitespace_partial.mustache new file mode 100644 index 0000000..ce43cb3 --- /dev/null +++ b/spec/_files/whitespace_partial.mustache @@ -0,0 +1,3 @@ +

{{ greeting }}

+{{> partial }} +

{{ farewell }}

\ No newline at end of file diff --git a/spec/_files/whitespace_partial.txt b/spec/_files/whitespace_partial.txt new file mode 100644 index 0000000..160b0b6 --- /dev/null +++ b/spec/_files/whitespace_partial.txt @@ -0,0 +1,6 @@ +

Welcome

+Hello Chris +You have just won $10000! +Well, $6000, after taxes. + +

Fair enough, right?

diff --git a/spec/mustache_spec.rb b/spec/mustache_spec.rb new file mode 100644 index 0000000..e04bec1 --- /dev/null +++ b/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 diff --git a/wrappers/commonjs/mustache.js.tpl.post b/wrappers/commonjs/mustache.js.tpl.post new file mode 100644 index 0000000..f566443 --- /dev/null +++ b/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); + }; +} diff --git a/wrappers/commonjs/mustache.js.tpl.pre b/wrappers/commonjs/mustache.js.tpl.pre new file mode 100644 index 0000000..5e33b34 --- /dev/null +++ b/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. + */ + diff --git a/wrappers/commonjs/package.json b/wrappers/commonjs/package.json new file mode 100644 index 0000000..8e13993 --- /dev/null +++ b/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" +} diff --git a/wrappers/dojo/mustache.js.tpl.post b/wrappers/dojo/mustache.js.tpl.post new file mode 100644 index 0000000..d64667f --- /dev/null +++ b/wrappers/dojo/mustache.js.tpl.post @@ -0,0 +1,4 @@ + + dojox.mustache = dojo.hitch(Mustache, "to_html"); + +})(); \ No newline at end of file diff --git a/wrappers/dojo/mustache.js.tpl.pre b/wrappers/dojo/mustache.js.tpl.pre new file mode 100644 index 0000000..f87f3cd --- /dev/null +++ b/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(){ diff --git a/wrappers/jquery/jquery.mustache.js.tpl.post b/wrappers/jquery/jquery.mustache.js.tpl.post new file mode 100644 index 0000000..1fe2887 --- /dev/null +++ b/wrappers/jquery/jquery.mustache.js.tpl.post @@ -0,0 +1,6 @@ + + $.mustache = function(template, view, partials) { + return Mustache.to_html(template, view, partials); + }; + +})(jQuery); diff --git a/wrappers/jquery/jquery.mustache.js.tpl.pre b/wrappers/jquery/jquery.mustache.js.tpl.pre new file mode 100644 index 0000000..b4d8af5 --- /dev/null +++ b/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($) { + diff --git a/wrappers/qooxdoo/qooxdoo.mustache.js.tpl.post b/wrappers/qooxdoo/qooxdoo.mustache.js.tpl.post new file mode 100644 index 0000000..e66f9e6 --- /dev/null +++ b/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; + +})(); diff --git a/wrappers/qooxdoo/qooxdoo.mustache.js.tpl.pre b/wrappers/qooxdoo/qooxdoo.mustache.js.tpl.pre new file mode 100644 index 0000000..f7d1014 --- /dev/null +++ b/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: + *
+ * var template = "Hi, my name is {{name}}!";
+ * var view = {name: "qooxdoo"};
+ * qx.bom.Template.toHtml(template, view);
+ * // return "Hi, my name is qooxdoo!"
+ * 
+ * + * 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. + */ + \ No newline at end of file diff --git a/wrappers/requirejs/requirejs.mustache.js.tpl.post b/wrappers/requirejs/requirejs.mustache.js.tpl.post new file mode 100644 index 0000000..632fd2a --- /dev/null +++ b/wrappers/requirejs/requirejs.mustache.js.tpl.post @@ -0,0 +1,3 @@ + +return Mustache; +}); \ No newline at end of file diff --git a/wrappers/requirejs/requirejs.mustache.js.tpl.pre b/wrappers/requirejs/requirejs.mustache.js.tpl.pre new file mode 100644 index 0000000..160286d --- /dev/null +++ b/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(){ diff --git a/wrappers/yui3/mustache.js.tpl.post b/wrappers/yui3/mustache.js.tpl.post new file mode 100644 index 0000000..373d827 --- /dev/null +++ b/wrappers/yui3/mustache.js.tpl.post @@ -0,0 +1,4 @@ + + Y.mustache = Mustache.to_html; + +}, "0"); diff --git a/wrappers/yui3/mustache.js.tpl.pre b/wrappers/yui3/mustache.js.tpl.pre new file mode 100644 index 0000000..280de86 --- /dev/null +++ b/wrappers/yui3/mustache.js.tpl.pre @@ -0,0 +1 @@ +YUI.add("mustache", function(Y) {