Compare commits

..

99 Commits

Author SHA1 Message Date
Justin Li
27507d087e Bump version to 2.6.3 2015-07-23 17:00:30 -04:00
Justin Li
0d0ece6ece Update history for 2.6.3 2015-07-23 16:59:44 -04:00
Dylan Thacker-Smith
435babc051 Fix a timezone test failure. 2015-07-23 16:51:20 -04:00
Arthur Neves
4f33cd1eae Bump to version 2.6.2 2015-01-23 10:18:03 -05:00
Florian Weingarten
1854cd17ab Bump version to 2.6.1 2015-01-23 10:17:18 -05:00
Arthur Nogueira Neves
69c2575485 Merge pull request #503 from parkr/patch-1
Remove duplicate `index0` key in htmltags
2014-12-30 13:15:04 -05:00
Parker Moore
0e98b29665 Remove duplicate index0 key in htmltags
Half of #502.
2014-12-26 16:33:47 -05:00
Florian Weingarten
deeb813d53 Merge pull request #300 from Shopify/cherry_pick_security_fix_to_2-6-stable
Cherry pick security fix to 2-6-stable
2014-01-10 10:20:41 -08:00
Florian Weingarten
eb409ff237 Cherry pick security fix (#274) to 2-6-stable 2014-01-10 11:22:28 -05:00
Arthur Neves
442041206f Update history log 2013-11-25 10:25:39 -05:00
Arthur Neves
dc6c6fbb9a Bump version to 2.6.0 2013-11-25 10:24:30 -05:00
Florian Weingarten
4293be3154 Update history 2013-11-11 09:02:32 -05:00
Arthur Nogueira Neves
736c11c876 Merge pull request #268 from Shopify/better_rake
Improve Rakefile
Conflicts:
	Rakefile
2013-10-10 10:59:26 -04:00
Arthur Neves
a41a60d294 Bump version to rc1 2013-10-10 10:22:35 -04:00
Florian Weingarten
307cab2eaa History and gemspec
Conflicts:
	History.md
	liquid.gemspec
2013-10-09 17:04:47 -04:00
Arthur Neves
a64360d148 Update history
Conflicts:
	History.md
2013-10-09 16:33:44 -04:00
Florian Weingarten
6cb5a9b7cc Merge pull request #234 from Shopify/fix_mapping_procs
Fix mapping over procs
2013-10-09 16:32:53 -04:00
Florian Weingarten
4207d1f086 Merge pull request #232 from Shopify/to_liquid_stuff
Always call 'to_liquid' on stuff in map filter and allow to_liquid to be...
2013-10-09 16:32:45 -04:00
Florian Weingarten
988a1694fd Use invoke_drop in map filter 2013-10-09 16:32:38 -04:00
Florian Weingarten
5c5e7de31e Merge pull request #221 from d-Pixie/master
Changes not empty? to any?
2013-07-01 07:55:26 -07:00
Jonas Schubert Erlandsson
f91233450f Changes not empty? to any? 2013-07-01 16:12:58 +02:00
Florian Weingarten
429e492984 Merge pull request #220 from coding46/master
Fix some typos in comments
2013-06-27 13:20:24 -07:00
G. Bodenschatz
d36a1c518b Fix some typos in comments 2013-06-27 22:16:23 +02:00
Florian Weingarten
37309678de Merge pull request #219 from Shopify/indent_and_trailing_ws_cleanup
Convert legacy tab indentation to spaces and remove trailing whitespace ...
2013-06-27 06:50:28 -07:00
Florian Weingarten
cfb60c2f1b Update History.md 2013-06-27 14:30:37 +02:00
Florian Weingarten
0f5441b09e Convert legacy tab indentation to spaces and remove trailing whitespace from all lines 2013-06-26 19:53:09 -04:00
Florian Weingarten
7a3746ad77 Update CONTRIBUTING.md
Add @arthurnn for code review
2013-06-26 05:03:05 +03:00
Florian Weingarten
24511556d3 Update README.md 2013-06-20 09:53:21 -03:00
Florian Weingarten
5621556b3a Update History.md 2013-06-18 12:36:50 -03:00
Florian Weingarten
fd230bef14 Merge pull request #106 from gnowoel/fix_example_servlet
fix example servlet
2013-06-18 08:35:42 -07:00
Florian Weingarten
e9f3a8e4d3 Update History.md 2013-06-18 05:18:43 +03:00
Florian Weingarten
8ca4868bff Merge branch 'master' of https://github.com/ndwebgroup/liquid into ndwebgroup-master 2013-06-18 04:13:49 +02:00
Florian Weingarten
f92da6948d Merge branch 'strip-html-fix' of https://github.com/jamesallardice/liquid into jamesallardice-strip-html-fix
Conflicts:
	lib/liquid/standardfilters.rb
2013-06-18 04:11:11 +02:00
Florian Weingarten
a1b156f0d4 strip_html multi-line comment test 2013-06-18 04:06:35 +02:00
Florian Weingarten
eca520025c Merge branch 'strip-html' of https://github.com/joliss/liquid into joliss-strip-html
Conflicts:
	lib/liquid/standardfilters.rb
2013-06-18 04:04:49 +02:00
Florian Weingarten
9c9a7ce8d3 Update History.md 2013-06-18 05:01:23 +03:00
Florian Weingarten
b7837ce218 Merge pull request #210 from Shopify/truncate_utf8_test
UTF8 truncate test
2013-06-17 09:45:10 -07:00
Florian Weingarten
b81469d183 Make truncate work for Ruby 1.8 2013-06-17 12:05:02 -04:00
Florian Weingarten
f488058789 UTF8 truncate test 2013-06-17 11:48:03 -04:00
Florian Weingarten
f5d75718e9 Update History.md 2013-06-16 18:31:19 +03:00
Florian Weingarten
c280936afa Merge pull request #209 from Shopify/fix_broken_raw_parsing
Fix broken 'raw' tag parsing (issue #204)
2013-06-15 04:58:51 -07:00
Florian Weingarten
e47d1af03a Fix broken 'raw' tag parsing (issue #204) 2013-06-14 12:47:27 -04:00
Florian Weingarten
81f6c79c53 Update README.md 2013-06-14 10:04:24 -03:00
Florian Weingarten
3174411407 Merge pull request #208 from Shopify/contributing
CONTRIBUTING.md
2013-06-14 05:58:37 -07:00
Florian Weingarten
f7c4b0cdec Update History.md 2013-06-14 08:57:27 -04:00
Florian Weingarten
3d1f582318 Merge pull request #201 from arthurnn/adding_version_file
create version.rb file, and bump version
2013-06-14 05:45:48 -07:00
Florian Weingarten
27db1def63 CONTRIBUTING.md 2013-06-14 02:41:08 +02:00
Florian Weingarten
5cfa13d7a4 Merge pull request #205 from phoet/stricter_handling_of_variables_in_for
rejects variables like a/b in for loops, closes #150
2013-06-13 15:26:00 -07:00
Peter Schröder
cb12497859 add test for allowing whitespace between tokens 2013-06-13 18:21:49 -04:00
Florian Weingarten
4c2e2f8a24 Merge pull request #203 from phoet/strip_carriage_return
handle carriage return in strip_newline, closes #126
2013-06-13 12:54:17 -07:00
Peter Schröder
a2df5a421d rejects variables like a/b in for loops, closes #150 2013-06-11 17:22:33 -04:00
Dylan Thacker-Smith
437e9201de Merge pull request #174 from yardstick/drop-context
Allow a Liquid::Drop to be passed into Template#render
2013-06-10 07:46:06 -07:00
Peter Schröder
fd263bba0f handle carriage return in strip_newline, closes #126 2013-06-09 10:24:35 -04:00
Arthur Neves
0d9353591b bump master version to 2.6.0
Bumping master version to 2.6.0
Adding 2.6.0 changes to history
2013-06-06 14:17:12 -04:00
Arthur Neves
1849c24f2c Adding version file
Follow the standard structure of having a version.rb file
2013-06-06 14:10:45 -04:00
Daniel Huckstep
f8288546f8 Missed in the rebase conflicts 2013-06-06 10:20:34 -06:00
Daniel Huckstep
076ae903c0 Make sure the context gets set 2013-06-06 10:19:20 -06:00
Daniel Huckstep
ba5e65f685 Better test, resuse Hash block 2013-06-06 10:18:47 -06:00
Daniel Huckstep
b699c93bae Allow a Liquid::Drop to be passed into Template#render 2013-06-06 10:17:01 -06:00
Florian Weingarten
85c1503378 Merge pull request #200 from arthurnn/155_float_precision
use BigDecimal on filters to have better precision
2013-06-05 14:07:20 -07:00
Arthur Neves
ab760649ee use BigDecimal on filters to have better precision 2013-06-05 16:09:05 -04:00
Florian Weingarten
f4fb2f159c Merge pull request #157 from stomar/avoid_warnings
Avoid warnings in Ruby 1.9.3
2013-06-05 09:10:41 -07:00
Florian Weingarten
a5cd494717 Merge pull request #176 from unreal/master
Add reverse filter
2013-06-05 09:04:53 -07:00
Florian Weingarten
f8915f3cd2 Merge pull request #199 from phoet/add_documentation_for_include
add documentation to include, fixes #163
2013-06-05 09:00:11 -07:00
Peter Schröder
e92540a9bf add documentation to include, fixes #163 2013-06-05 11:53:01 -04:00
Dylan Thacker-Smith
482115d784 Merge pull request #173 from jsw0528/master
fix `can't convert Fixnum into String` for `replace`
2013-06-04 13:09:42 -07:00
Florian Weingarten
94ff457744 Merge pull request #198 from Shopify/limit_resource_usage
Resource usage limits
2013-05-31 08:24:29 -07:00
Florian Weingarten
1e8c081b42 Create new resource_limits hash on Template initialization 2013-05-31 09:41:59 -04:00
Florian Weingarten
2b17e24b16 Mutate resource_limits hash to flag that the limit was reached (for outside observation) 2013-05-31 09:34:23 -04:00
Florian Weingarten
9075b428b1 Resource limits: Don't raise Error but render error message (but abort after first error) 2013-05-31 09:25:25 -04:00
Florian Weingarten
8760b5e8c4 Add optional resource usage limitations to number of rendering calls, length of rendering output and/or number of variable/capture assignments 2013-05-30 17:04:26 -04:00
Tom Burns
50b2ebee56 Merge pull request #189 from Shopify/cache_partials
Cache tokenized partial templates
2013-05-29 07:51:12 -07:00
Dylan Thacker-Smith
23203c0122 Fix some old templates that abused colon as an argument separator.
This is a fallback for keyword argument parsing since this feature broke
old templates that accidentally used a colon as a filter argument
separator.
2013-05-21 17:47:46 -04:00
Tom Burns
27fe76c0dd Merge pull request #192 from Shopify/revert_utf8
Revert "Merge pull request #185 from ISSIntel/liquid-utf8"
2013-05-21 14:46:17 -07:00
Tom Burns
8913a5615a Revert "Merge pull request #185 from ISSIntel/liquid-utf8"
This reverts commit c5dfcd29b0, reversing
changes made to f7d1e1d0c1.
2013-05-20 19:53:13 -04:00
Dylan Thacker-Smith
690b3ff27f Merge pull request #135 from astathopoulos/preserve_filters_ordering
Use array instead of Hash to keep the registered filters
2013-05-17 05:25:07 -07:00
Tasos Stathopoulos
8c1bbfec57 Use array instead of Hash to keep the registered filters
1.8.7 compatibility fix

In Ruby 1.8.7, Hash does not preserve insertion ordering as Array does.
This could cause a problem when registering filters which depend on others and
the registration order is important.

So, the @@filters variable was changed to array where the order of the filters is
the same as the insertion order.
2013-05-17 14:12:57 +03:00
Tom Burns
b8fbd2b4fa typo 2013-05-16 20:25:31 -04:00
Tom Burns
ba5a9f2e47 remove _ on private methods 2013-05-13 13:45:43 -04:00
Tom Burns
1e309ba74b cache included partial templates 2013-05-13 02:34:19 -04:00
Tom Burns
485340713a Add tests for caching partial includes 2013-05-13 02:34:19 -04:00
Tom Burns
2af4ea1295 Support benchmarking templates with 'include' tag 2013-05-12 22:17:08 -04:00
Tom Burns
c5dfcd29b0 Merge pull request #185 from ISSIntel/liquid-utf8
Liquid UTF-8 support
2013-04-23 10:29:49 -07:00
Ozéias Sant'ana
f9c289372d Merge branch 'master' into liquid-utf8
Conflicts:
	lib/liquid/variable.rb
2013-04-23 13:10:16 -03:00
Jay Strybis
a556ae6c26 Add reverse filter 2013-03-07 16:29:39 -06:00
Marcus Stollsteimer
1cac09831d Completely remove unused variable 2013-03-05 22:19:38 +01:00
wǒ_is神仙
17dd85868d add tests for replace filter 2013-02-21 10:52:46 +08:00
wǒ_is神仙
6e967f7f3a fix can't convert Fixnum into String 2013-02-06 12:55:35 +08:00
Adam Tanner
0b36540b78 Liquid has UTF8 support. 2012-12-26 18:14:36 -08:00
Marcus Stollsteimer
b48ad7da3a Remove trailing whitespace 2012-11-18 10:29:22 +01:00
Marcus Stollsteimer
afc3944a4a Fix assignment with no effect outside of iterator 2012-11-18 10:21:03 +01:00
Marcus Stollsteimer
c79abf1f87 Avoid warnings for assigned but unused variable 2012-11-18 10:20:07 +01:00
Marcus Stollsteimer
90b40ffb4b Avoid warnings for shadowed outer local variable 2012-11-18 10:19:05 +01:00
Marcus Stollsteimer
fea9c54768 Avoid warning for grouped expression 2012-11-18 10:08:00 +01:00
Jeremy Friesen
740cd6e762 Merge branch 'master' of git://github.com/Shopify/liquid
* 'master' of git://github.com/Shopify/liquid:
  * Seperated 'Howto' into 'How to'. * Added periods to the second list as the first item has them. I guess I'm anally retentive like that. :)
  Fix conditions using negative number comparisons
2012-04-19 10:59:44 -04:00
Jeremy Friesen
5d0004a87e Added tests to verify that {{ 'now' | date :'%Y' }} and {{ 'today' | date :'%Y' }} work.
In the case of Ruby 1.9.3, 'now' is no longer parsed.  For safe
measures, I've added 'today' as well.
2012-04-19 10:48:55 -04:00
jamesallardice
12112dee35 Make strip_html filter strip contents of style tags 2012-04-11 16:23:21 +01:00
Leo Wong
5c4938f443 fix example servlet 2012-03-08 01:29:50 +08:00
Jo Liss
4a2bbafeb4 Make strip_html strip tags spread across lines 2012-02-16 15:42:10 +01:00
95 changed files with 2429 additions and 2077 deletions

1
.gitignore vendored
View File

@@ -4,3 +4,4 @@
pkg pkg
*.rbc *.rbc
.rvmrc .rvmrc
.ruby-version

26
CONTRIBUTING.md Normal file
View File

@@ -0,0 +1,26 @@
# How to contribute
## Things we will merge
* Bugfixes
* Performance improvements
* Features which are likely to be useful to the majority of Liquid users
## Things we won't merge
* Code which introduces considerable performance degrations
* Code which touches performance critical parts of Liquid and comes without benchmarks
* Features which are not important for most people (we want to keep the core Liquid code small and tidy)
* Features which can easily be implemented on top of Liquid (for example as a custom filter or custom filesystem)
* Code which comes without tests
* Code which breaks existing tests
## Workflow
* Fork the Liquid repository
* Create a new branch in your fork
* If it makes sense, add tests for your code and run a performance benchmark
* Make sure all tests pass
* Create a pull request
* In the description, ping one of [@boourns](https://github.com/boourns), [@fw42](https://github.com/fw42), [@camilo](https://github.com/camilo), [@dylanahsmith](https://github.com/dylanahsmith), or [@arthurnn](https://github.com/arthurnn) and ask for a code review.

View File

@@ -1,12 +1,45 @@
# Liquid Version History # Liquid Version History
## 2.5.5 / 2014-01-10 / branch "2-5-stable" IMPORTANT: Liquid 2.6 is going to be the last version of Liquid which maintains explicit Ruby 1.8 compatability.
The following releases will only be tested against Ruby 1.9 and Ruby 2.0 and are likely to break on Ruby 1.8.
## 2.6.3 / 2015-07-23 / branch "2-6-stable"
* Fix test failure under certain timezones [Dylan Thacker-Smith]
## 2.6.2 / 2015-01-23
* Remove duplicate hash key [Parker Moore]
## 2.6.1 / 2014-01-10
Security fix, cherry-picked from master (4e14a65): Security fix, cherry-picked from master (4e14a65):
* Don't call to_sym when creating conditions for security reasons, see #273 [Bouke van der Bijl, bouk] * Don't call to_sym when creating conditions for security reasons, see #273 [Bouke van der Bijl, bouk]
* Prevent arbitrary method invocation on condition objects, see #274 [Dylan Thacker-Smith, dylanahsmith] * Prevent arbitrary method invocation on condition objects, see #274 [Dylan Thacker-Smith, dylanahsmith]
## 2.5.4 / 2013-11-11 ## 2.6.0 / 2013-11-25
* ...
* Bugfix for #106: fix example servlet [gnowoel]
* Bugfix for #97: strip_html filter supports multi-line tags [Jo Liss, joliss]
* Bugfix for #114: strip_html filter supports style tags [James Allardice, jamesallardice]
* Bugfix for #117: 'now' support for date filter in Ruby 1.9 [Notre Dame Webgroup, ndwebgroup]
* Bugfix for #166: truncate filter on UTF-8 strings with Ruby 1.8 [Florian Weingarten, fw42]
* Bugfix for #204: 'raw' parsing bug [Florian Weingarten, fw42]
* Bugfix for #150: 'for' parsing bug [Peter Schröder, phoet]
* Bugfix for #126: Strip CRLF in strip_newline [Peter Schröder, phoet]
* Bugfix for #174, "can't convert Fixnum into String" for "replace" [wǒ_is神仙, jsw0528]
* Allow a Liquid::Drop to be passed into Template#render [Daniel Huckstep, darkhelmet]
* Resource limits [Florian Weingarten, fw42]
* Add reverse filter [Jay Strybis, unreal]
* Add utf-8 support
* Use array instead of Hash to keep the registered filters [Tasos Stathopoulos, astathopoulos]
* Cache tokenized partial templates [Tom Burns, boourns]
* Avoid warnings in Ruby 1.9.3 [Marcus Stollsteimer, stomar]
* Better documentation for 'include' tag (closes #163) [Peter Schröder, phoet]
* Use of BigDecimal on filters to have better precision (closes #155) [Arthur Nogueira Neves, arthurnn]
## 2.5.4 / 2013-11-11 / branch "2.5-stable"
* Fix "can't convert Fixnum into String" for "replace", see #173, [wǒ_is神仙, jsw0528] * Fix "can't convert Fixnum into String" for "replace", see #173, [wǒ_is神仙, jsw0528]
@@ -30,6 +63,7 @@ Yanked from rubygems, as it contained too many changes that broke compatibility.
* Fix filter parser for args without space separators * Fix filter parser for args without space separators
* Add support for filter keyword arguments * Add support for filter keyword arguments
## 2.4.0 / 2012-08-03 ## 2.4.0 / 2012-08-03
* Performance improvements * Performance improvements

View File

@@ -17,4 +17,4 @@ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

View File

@@ -1,5 +1,11 @@
# Liquid template engine # Liquid template engine
* [Contributing guidelines](CONTRIBUTING.md)
* [Version history](History.md)
* [Liquid documentation from Shopify](http://docs.shopify.com/themes/liquid-basics)
* [Liquid Wiki from Shopify](http://wiki.shopify.com/Liquid)
* [Website](http://liquidmarkup.org/)
## Introduction ## Introduction
Liquid is a template engine which was written with very specific requirements: Liquid is a template engine which was written with very specific requirements:

View File

@@ -1,9 +1,7 @@
#!/usr/bin/env ruby
require 'rubygems'
require 'rake' require 'rake'
require 'rake/testtask' require 'rake/testtask'
require 'rubygems/package_task' $LOAD_PATH.unshift File.expand_path("../lib", __FILE__)
require "liquid/version"
task :default => 'test' task :default => 'test'
@@ -13,14 +11,20 @@ Rake::TestTask.new(:test) do |t|
t.verbose = false t.verbose = false
end end
gemspec = eval(File.read('liquid.gemspec')) task :gem => :build
Gem::PackageTask.new(gemspec) do |pkg| task :build do
pkg.gem_spec = gemspec system "gem build liquid.gemspec"
end end
desc "Build the gem and release it to rubygems.org" task :install => :build do
task :release => :gem do system "gem install liquid-#{Liquid::VERSION}.gem"
sh "gem push pkg/liquid-#{gemspec.version}.gem" end
task :release => :build do
system "git tag -a v#{Liquid::VERSION} -m 'Tagging #{Liquid::VERSION}'"
system "git push --tags"
system "gem push liquid-#{Liquid::VERSION}.gem"
system "rm liquid-#{Liquid::VERSION}.gem"
end end
namespace :benchmark do namespace :benchmark do

View File

@@ -2,40 +2,40 @@ module ProductsFilter
def price(integer) def price(integer)
sprintf("$%.2d USD", integer / 100.0) sprintf("$%.2d USD", integer / 100.0)
end end
def prettyprint(text) def prettyprint(text)
text.gsub( /\*(.*)\*/, '<b>\1</b>' ) text.gsub( /\*(.*)\*/, '<b>\1</b>' )
end end
def count(array) def count(array)
array.size array.size
end end
def paragraph(p) def paragraph(p)
"<p>#{p}</p>" "<p>#{p}</p>"
end end
end end
class Servlet < LiquidServlet class Servlet < LiquidServlet
def index def index
{ 'date' => Time.now } { 'date' => Time.now }
end end
def products def products
{ 'products' => products_list, 'section' => 'Snowboards', 'cool_products' => true} { 'products' => products_list, 'description' => description, 'section' => 'Snowboards', 'cool_products' => true}
end
private
def products_list
[{'name' => 'Arbor Draft', 'price' => 39900, 'description' => 'the *arbor draft* is a excellent product' },
{'name' => 'Arbor Element', 'price' => 40000, 'description' => 'the *arbor element* rocks for freestyling'},
{'name' => 'Arbor Diamond', 'price' => 59900, 'description' => 'the *arbor diamond* is a made up product because im obsessed with arbor and have no creativity'}]
end end
def description def description
"List of Products ~ This is a list of products with price and description." "List of Products ~ This is a list of products with price and description."
end end
private
def products_list
[{'name' => 'Arbor Draft', 'price' => 39900, 'description' => 'the *arbor draft* is a excellent product' },
{'name' => 'Arbor Element', 'price' => 40000, 'description' => 'the *arbor element* rocks for freestyling'},
{'name' => 'Arbor Diamond', 'price' => 59900, 'description' => 'the *arbor diamond* is a made up product because im obsessed with arbor and have no creativity'}]
end
end end

View File

@@ -7,22 +7,22 @@ class LiquidServlet < WEBrick::HTTPServlet::AbstractServlet
def do_POST(req, res) def do_POST(req, res)
handle(:post, req, res) handle(:post, req, res)
end end
private private
def handle(type, req, res) def handle(type, req, res)
@request, @response = req, res @request, @response = req, res
@request.path_info =~ /(\w+)$/ @request.path_info =~ /(\w+)$/
@action = $1 || 'index' @action = $1 || 'index'
@assigns = send(@action) if respond_to?(@action) @assigns = send(@action) if respond_to?(@action)
@response['Content-Type'] = "text/html" @response['Content-Type'] = "text/html"
@response.status = 200 @response.status = 200
@response.body = Liquid::Template.parse(read_template).render(@assigns, :filters => [ProductsFilter]) @response.body = Liquid::Template.parse(read_template).render(@assigns, :filters => [ProductsFilter])
end end
def read_template(filename = @action) def read_template(filename = @action)
File.read( File.dirname(__FILE__) + "/templates/#{filename}.liquid" ) File.read( File.dirname(__FILE__) + "/templates/#{filename}.liquid" )
end end
end end

View File

@@ -3,4 +3,4 @@
<p>It is {{date}}</p> <p>It is {{date}}</p>
<p>Check out the <a href="http://localhost:3000/products">Products</a> screen </p> <p>Check out the <a href="http://localhost:3000/products">Products</a> screen </p>

View File

@@ -1,49 +1,49 @@
<?xml version="1.0" encoding="utf-8"?> <?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en"> <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head> <head>
<meta http-equiv="Content-type" content="text/html; charset=utf-8" /> <meta http-equiv="Content-type" content="text/html; charset=utf-8" />
<meta http-equiv="Content-Language" content="en-us" /> <meta http-equiv="Content-Language" content="en-us" />
<title>products</title>
<meta name="ROBOTS" content="ALL" />
<meta http-equiv="imagetoolbar" content="no" />
<meta name="MSSmartTagsPreventParsing" content="true" />
<meta name="Copyright" content="(c) 2005 Copyright content: Copyright design: Tobias Luetke" />
<!-- (c) Copyright 2005 by Tobias Luetke All Rights Reserved. -->
</head>
<body>
<h1>{{ description | split: '~' | first }}</h1> <title>products</title>
<h2>{{ description | split: '~' | last }}</h2> <meta name="ROBOTS" content="ALL" />
<meta http-equiv="imagetoolbar" content="no" />
<meta name="MSSmartTagsPreventParsing" content="true" />
<meta name="Copyright" content="(c) 2005 Copyright content: Copyright design: Tobias Luetke" />
<!-- (c) Copyright 2005 by Tobias Luetke All Rights Reserved. -->
</head>
<h2>There are currently {{products | count}} products in the {{section}} catalog</h2> <body>
<h1>{{ description | split: '~' | first }}</h1>
<h2>{{ description | split: '~' | last }}</h2>
<h2>There are currently {{products | count}} products in the {{section}} catalog</h2>
{% if cool_products %} {% if cool_products %}
Cool products :) Cool products :)
{% else %} {% else %}
Uncool products :( Uncool products :(
{% endif %} {% endif %}
<ul id="products"> <ul id="products">
{% for product in products %} {% for product in products %}
<li> <li>
<h2>{{product.name}}</h2> <h2>{{product.name}}</h2>
Only {{product.price | price }} Only {{product.price | price }}
{{product.description | prettyprint | paragraph }} {{product.description | prettyprint | paragraph }}
{{ 'it rocks!' | paragraph }} {{ 'it rocks!' | paragraph }}
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
</body> </body>
</html> </html>

View File

@@ -2,15 +2,15 @@
# and use liquid as an template system for .liquid files # and use liquid as an template system for .liquid files
# #
# Example # Example
# #
# ActionView::Base::register_template_handler :liquid, LiquidView # ActionView::Base::register_template_handler :liquid, LiquidView
class LiquidView class LiquidView
PROTECTED_ASSIGNS = %w( template_root response _session template_class action_name request_origin session template PROTECTED_ASSIGNS = %w( template_root response _session template_class action_name request_origin session template
_response url _request _cookies variables_added _flash params _headers request cookies _response url _request _cookies variables_added _flash params _headers request cookies
ignore_missing_templates flash _params logger before_filter_chain_aborted headers ) ignore_missing_templates flash _params logger before_filter_chain_aborted headers )
PROTECTED_INSTANCE_VARIABLES = %w( @_request @controller @_first_render @_memoized__pick_template @view_paths PROTECTED_INSTANCE_VARIABLES = %w( @_request @controller @_first_render @_memoized__pick_template @view_paths
@helpers @assigns_added @template @_render_stack @template_format @assigns ) @helpers @assigns_added @template @_render_stack @template_format @assigns )
def self.call(template) def self.call(template)
"LiquidView.new(self).render(template, local_assigns)" "LiquidView.new(self).render(template, local_assigns)"
end end
@@ -18,10 +18,10 @@ class LiquidView
def initialize(view) def initialize(view)
@view = view @view = view
end end
def render(template, local_assigns = nil) def render(template, local_assigns = nil)
@view.controller.headers["Content-Type"] ||= 'text/html; charset=utf-8' @view.controller.headers["Content-Type"] ||= 'text/html; charset=utf-8'
# Rails 2.2 Template has source, but not locals # Rails 2.2 Template has source, but not locals
if template.respond_to?(:source) && !template.respond_to?(:locals) if template.respond_to?(:source) && !template.respond_to?(:locals)
assigns = (@view.instance_variables - PROTECTED_INSTANCE_VARIABLES).inject({}) do |hash, ivar| assigns = (@view.instance_variables - PROTECTED_INSTANCE_VARIABLES).inject({}) do |hash, ivar|
@@ -31,15 +31,15 @@ class LiquidView
else else
assigns = @view.assigns.reject{ |k,v| PROTECTED_ASSIGNS.include?(k) } assigns = @view.assigns.reject{ |k,v| PROTECTED_ASSIGNS.include?(k) }
end end
source = template.respond_to?(:source) ? template.source : template source = template.respond_to?(:source) ? template.source : template
local_assigns = (template.respond_to?(:locals) ? template.locals : local_assigns) || {} local_assigns = (template.respond_to?(:locals) ? template.locals : local_assigns) || {}
if content_for_layout = @view.instance_variable_get("@content_for_layout") if content_for_layout = @view.instance_variable_get("@content_for_layout")
assigns['content_for_layout'] = content_for_layout assigns['content_for_layout'] = content_for_layout
end end
assigns.merge!(local_assigns.stringify_keys) assigns.merge!(local_assigns.stringify_keys)
liquid = Liquid::Template.parse(source) liquid = Liquid::Template.parse(source)
liquid.render(assigns, :filters => [@view.controller.master_helper_module], :registers => {:action_view => @view, :controller => @view.controller}) liquid.render(assigns, :filters => [@view.controller.master_helper_module], :registers => {:action_view => @view, :controller => @view.controller})
end end
@@ -48,4 +48,4 @@ class LiquidView
false false
end end
end end

View File

@@ -45,6 +45,7 @@ module Liquid
VariableParser = /\[[^\]]+\]|#{VariableSegment}+\??/o VariableParser = /\[[^\]]+\]|#{VariableSegment}+\??/o
end end
require "liquid/version"
require 'liquid/drop' require 'liquid/drop'
require 'liquid/extensions' require 'liquid/extensions'
require 'liquid/errors' require 'liquid/errors'

View File

@@ -16,7 +16,7 @@ module Liquid
when IsTag when IsTag
if token =~ FullToken if token =~ FullToken
# if we found the proper block delimitor just end parsing here and let the outer block # if we found the proper block delimiter just end parsing here and let the outer block
# proceed # proceed
if block_delimiter == $1 if block_delimiter == $1
end_tag end_tag
@@ -43,8 +43,8 @@ module Liquid
end end
end end
# Make sure that its ok to end parsing in the current block. # Make sure that it's ok to end parsing in the current block.
# Effectively this method will throw and exception unless the current block is # Effectively this method will throw an exception unless the current block is
# of type Document # of type Document
assert_missing_delimitation! assert_missing_delimitation!
end end
@@ -90,20 +90,31 @@ module Liquid
def render_all(list, context) def render_all(list, context)
output = [] output = []
context.resource_limits[:render_length_current] = 0
context.resource_limits[:render_score_current] += list.length
list.each do |token| list.each do |token|
# Break out if we have any unhanded interrupts. # Break out if we have any unhanded interrupts.
break if context.has_interrupt? break if context.has_interrupt?
begin begin
# If we get an Interrupt that means the block must stop processing. An # If we get an Interrupt that means the block must stop processing. An
# Interrupt is any command that stops block execution such as {% break %} # Interrupt is any command that stops block execution such as {% break %}
# or {% continue %} # or {% continue %}
if token.is_a? Continue or token.is_a? Break if token.is_a? Continue or token.is_a? Break
context.push_interrupt(token.interrupt) context.push_interrupt(token.interrupt)
break break
end end
output << (token.respond_to?(:render) ? token.render(context) : token) token_output = (token.respond_to?(:render) ? token.render(context) : token)
context.resource_limits[:render_length_current] += (token_output.respond_to?(:length) ? token_output.length : 1)
if context.resource_limits_reached?
context.resource_limits[:reached] = true
raise MemoryError.new("Memory limits exceeded")
end
output << token_output
rescue MemoryError => e
raise e
rescue ::StandardError => e rescue ::StandardError => e
output << (context.handle_error(e)) output << (context.handle_error(e))
end end

View File

@@ -13,19 +13,26 @@ module Liquid
# #
# context['bob'] #=> nil class Context # context['bob'] #=> nil class Context
class Context class Context
attr_reader :scopes, :errors, :registers, :environments attr_reader :scopes, :errors, :registers, :environments, :resource_limits
def initialize(environments = {}, outer_scope = {}, registers = {}, rethrow_errors = false) def initialize(environments = {}, outer_scope = {}, registers = {}, rethrow_errors = false, resource_limits = {})
@environments = [environments].flatten @environments = [environments].flatten
@scopes = [(outer_scope || {})] @scopes = [(outer_scope || {})]
@registers = registers @registers = registers
@errors = [] @errors = []
@rethrow_errors = rethrow_errors @rethrow_errors = rethrow_errors
@resource_limits = (resource_limits || {}).merge!({ :render_score_current => 0, :assign_score_current => 0 })
squash_instance_assigns_with_environments squash_instance_assigns_with_environments
@interrupts = [] @interrupts = []
end end
def resource_limits_reached?
(@resource_limits[:render_length_limit] && @resource_limits[:render_length_current] > @resource_limits[:render_length_limit]) ||
(@resource_limits[:render_score_limit] && @resource_limits[:render_score_current] > @resource_limits[:render_score_limit] ) ||
(@resource_limits[:assign_score_limit] && @resource_limits[:assign_score_current] > @resource_limits[:assign_score_limit] )
end
def strainer def strainer
@strainer ||= Strainer.create(self) @strainer ||= Strainer.create(self)
end end
@@ -46,7 +53,7 @@ module Liquid
# are there any not handled interrupts? # are there any not handled interrupts?
def has_interrupt? def has_interrupt?
!@interrupts.empty? @interrupts.any?
end end
# push an interrupt to the stack. this interrupt is considered not handled. # push an interrupt to the stack. this interrupt is considered not handled.
@@ -165,6 +172,7 @@ module Liquid
# Fetches an object starting at the local scope and then moving up the hierachy # Fetches an object starting at the local scope and then moving up the hierachy
def find_variable(key) def find_variable(key)
scope = @scopes.find { |s| s.has_key?(key) } scope = @scopes.find { |s| s.has_key?(key) }
variable = nil
if scope.nil? if scope.nil?
@environments.each do |e| @environments.each do |e|

View File

@@ -5,7 +5,7 @@ module Liquid
parse(tokens) parse(tokens)
end end
# There isn't a real delimter # There isn't a real delimiter
def block_delimiter def block_delimiter
[] []
end end

View File

@@ -1,6 +1,6 @@
module Liquid module Liquid
class Error < ::StandardError; end class Error < ::StandardError; end
class ArgumentError < Error; end class ArgumentError < Error; end
class ContextError < Error; end class ContextError < Error; end
class FilterNotFound < Error; end class FilterNotFound < Error; end
@@ -8,4 +8,5 @@ module Liquid
class StandardError < Error; end class StandardError < Error; end
class SyntaxError < Error; end class SyntaxError < Error; end
class StackLevelError < Error; end class StackLevelError < Error; end
class MemoryError < Error; end
end end

View File

@@ -1,7 +1,7 @@
module Liquid module Liquid
# A Liquid file system is way to let your templates retrieve other templates for use with the include tag. # A Liquid file system is a way to let your templates retrieve other templates for use with the include tag.
# #
# You can implement subclasses that retrieve templates from the database, from the file system using a different # You can implement subclasses that retrieve templates from the database, from the file system using a different
# path structure, you can provide them as hard-coded inline strings, or any manner that you see fit. # path structure, you can provide them as hard-coded inline strings, or any manner that you see fit.
# #
# You can add additional instance variables, arguments, or methods as needed. # You can add additional instance variables, arguments, or methods as needed.
@@ -18,7 +18,7 @@ module Liquid
raise FileSystemError, "This liquid context does not allow includes." raise FileSystemError, "This liquid context does not allow includes."
end end
end end
# This implements an abstract file system which retrieves template files named in a manner similar to Rails partials, # This implements an abstract file system which retrieves template files named in a manner similar to Rails partials,
# ie. with the template name prefixed with an underscore. The extension ".liquid" is also added. # ie. with the template name prefixed with an underscore. The extension ".liquid" is also added.
# #
@@ -27,36 +27,36 @@ module Liquid
# Example: # Example:
# #
# file_system = Liquid::LocalFileSystem.new("/some/path") # file_system = Liquid::LocalFileSystem.new("/some/path")
# #
# file_system.full_path("mypartial") # => "/some/path/_mypartial.liquid" # file_system.full_path("mypartial") # => "/some/path/_mypartial.liquid"
# file_system.full_path("dir/mypartial") # => "/some/path/dir/_mypartial.liquid" # file_system.full_path("dir/mypartial") # => "/some/path/dir/_mypartial.liquid"
# #
class LocalFileSystem class LocalFileSystem
attr_accessor :root attr_accessor :root
def initialize(root) def initialize(root)
@root = root @root = root
end end
def read_template_file(template_path, context) def read_template_file(template_path, context)
full_path = full_path(template_path) full_path = full_path(template_path)
raise FileSystemError, "No such template '#{template_path}'" unless File.exists?(full_path) raise FileSystemError, "No such template '#{template_path}'" unless File.exists?(full_path)
File.read(full_path) File.read(full_path)
end end
def full_path(template_path) def full_path(template_path)
raise FileSystemError, "Illegal template name '#{template_path}'" unless template_path =~ /^[^.\/][a-zA-Z0-9_\/]+$/ raise FileSystemError, "Illegal template name '#{template_path}'" unless template_path =~ /^[^.\/][a-zA-Z0-9_\/]+$/
full_path = if template_path.include?('/') full_path = if template_path.include?('/')
File.join(root, File.dirname(template_path), "_#{File.basename(template_path)}.liquid") File.join(root, File.dirname(template_path), "_#{File.basename(template_path)}.liquid")
else else
File.join(root, "_#{template_path}.liquid") File.join(root, "_#{template_path}.liquid")
end end
raise FileSystemError, "Illegal template path '#{File.expand_path(full_path)}'" unless File.expand_path(full_path) =~ /^#{File.expand_path(root)}/ raise FileSystemError, "Illegal template path '#{File.expand_path(full_path)}'" unless File.expand_path(full_path) =~ /^#{File.expand_path(root)}/
full_path full_path
end end
end end
end end

View File

@@ -43,7 +43,6 @@ module Liquid
'index0' => index, 'index0' => index,
'col' => col + 1, 'col' => col + 1,
'col0' => col, 'col0' => col,
'index0' => index,
'rindex' => length - index, 'rindex' => length - index,
'rindex0' => length - index - 1, 'rindex0' => length - index - 1,
'first' => (index == 0), 'first' => (index == 0),
@@ -57,7 +56,7 @@ module Liquid
result << "<td class=\"col#{col}\">" << render_all(@nodelist, context) << '</td>' result << "<td class=\"col#{col}\">" << render_all(@nodelist, context) << '</td>'
if col == cols and not (index == length - 1) if col == cols and (index != length - 1)
col = 0 col = 0
row += 1 row += 1
result << "</tr>\n<tr class=\"row#{row}\">" result << "</tr>\n<tr class=\"row#{row}\">"

View File

@@ -2,7 +2,7 @@
# This library is free software. It may be used, redistributed and/or modified # This library is free software. It may be used, redistributed and/or modified
# under the same terms as Ruby itself # under the same terms as Ruby itself
# #
# This extension is usesd in order to expose the object of the implementing class # This extension is used in order to expose the object of the implementing class
# to liquid as it were a Drop. It also limits the liquid-callable methods of the instance # to liquid as it were a Drop. It also limits the liquid-callable methods of the instance
# to the allowed method passed with the liquid_methods call # to the allowed method passed with the liquid_methods call
# Example: # Example:

View File

@@ -1,4 +1,5 @@
require 'cgi' require 'cgi'
require 'bigdecimal'
module Liquid module Liquid
@@ -10,12 +11,12 @@ module Liquid
input.respond_to?(:size) ? input.size : 0 input.respond_to?(:size) ? input.size : 0
end end
# convert a input string to DOWNCASE # convert an input string to DOWNCASE
def downcase(input) def downcase(input)
input.to_s.downcase input.to_s.downcase
end end
# convert a input string to UPCASE # convert an input string to UPCASE
def upcase(input) def upcase(input)
input.to_s.upcase input.to_s.upcase
end end
@@ -42,7 +43,8 @@ module Liquid
if input.nil? then return end if input.nil? then return end
l = length.to_i - truncate_string.length l = length.to_i - truncate_string.length
l = 0 if l < 0 l = 0 if l < 0
input.length > length.to_i ? input[0...l] + truncate_string : input truncated = RUBY_VERSION[0,3] == "1.8" ? input.scan(/./mu)[0...l].to_s : input[0...l]
input.length > length.to_i ? truncated + truncate_string : input
end end
def truncatewords(input, words = 15, truncate_string = "...") def truncatewords(input, words = 15, truncate_string = "...")
@@ -63,15 +65,14 @@ module Liquid
end end
def strip_html(input) def strip_html(input)
input.to_s.gsub(/<script.*?<\/script>/, '').gsub(/<!--.*?-->/, '').gsub(/<.*?>/, '') input.to_s.gsub(/<script.*?<\/script>/m, '').gsub(/<!--.*?-->/m, '').gsub(/<style.*?<\/style>/m, '').gsub(/<.*?>/m, '')
end end
# Remove all newlines from the string # Remove all newlines from the string
def strip_newlines(input) def strip_newlines(input)
input.to_s.gsub(/\n/, '') input.to_s.gsub(/\r?\n/, '')
end end
# Join elements of the array with certain character between them # Join elements of the array with certain character between them
def join(input, glue = ' ') def join(input, glue = ' ')
[input].flatten.join(glue) [input].flatten.join(glue)
@@ -90,6 +91,12 @@ module Liquid
end end
end end
# Reverse the elements of an array
def reverse(input)
ary = [input].flatten
ary.reverse
end
# map/collect on a given property # map/collect on a given property
def map(input, property) def map(input, property)
ary = [input].flatten ary = [input].flatten
@@ -178,14 +185,23 @@ module Liquid
input = Time.at(input.to_i) input = Time.at(input.to_i)
end end
date = input.is_a?(String) ? Time.parse(input) : input date = if input.is_a?(String)
case input.downcase
when 'now', 'today'
Time.now
else
Time.parse(input)
end
else
input
end
if date.respond_to?(:strftime) if date.respond_to?(:strftime)
date.strftime(format.to_s) date.strftime(format.to_s)
else else
input input
end end
rescue => e rescue
input input
end end
@@ -209,41 +225,47 @@ module Liquid
# addition # addition
def plus(input, operand) def plus(input, operand)
to_number(input) + to_number(operand) apply_operation(input, operand, :+)
end end
# subtraction # subtraction
def minus(input, operand) def minus(input, operand)
to_number(input) - to_number(operand) apply_operation(input, operand, :-)
end end
# multiplication # multiplication
def times(input, operand) def times(input, operand)
to_number(input) * to_number(operand) apply_operation(input, operand, :*)
end end
# division # division
def divided_by(input, operand) def divided_by(input, operand)
to_number(input) / to_number(operand) apply_operation(input, operand, :/)
end end
def modulo(input, operand) def modulo(input, operand)
to_number(input) % to_number(operand) apply_operation(input, operand, :%)
end end
private private
def to_number(obj) def to_number(obj)
case obj case obj
when Numeric when Float
obj BigDecimal.new(obj.to_s)
when String when Numeric
(obj.strip =~ /^\d+\.\d+$/) ? obj.to_f : obj.to_i obj
else when String
0 (obj.strip =~ /^\d+\.\d+$/) ? BigDecimal.new(obj) : obj.to_i
end else
0
end end
end
def apply_operation(input, operand, operation)
result = to_number(input).send(operation, to_number(operand))
result.is_a?(BigDecimal) ? result.to_f : result
end
end end
Template.register_filter(StandardFilters) Template.register_filter(StandardFilters)

View File

@@ -8,7 +8,7 @@ module Liquid
# The Strainer only allows method calls defined in filters given to it via Strainer.global_filter, # The Strainer only allows method calls defined in filters given to it via Strainer.global_filter,
# Context#add_filters or Template.register_filter # Context#add_filters or Template.register_filter
class Strainer #:nodoc: class Strainer #:nodoc:
@@filters = {} @@filters = []
@@known_filters = Set.new @@known_filters = Set.new
@@known_methods = Set.new @@known_methods = Set.new
@@ -19,7 +19,7 @@ module Liquid
def self.global_filter(filter) def self.global_filter(filter)
raise ArgumentError, "Passed filter is not a module" unless filter.is_a?(Module) raise ArgumentError, "Passed filter is not a module" unless filter.is_a?(Module)
add_known_filter(filter) add_known_filter(filter)
@@filters[filter.name] = filter @@filters << filter unless @@filters.include?(filter)
end end
def self.add_known_filter(filter) def self.add_known_filter(filter)
@@ -34,7 +34,7 @@ module Liquid
def self.create(context) def self.create(context)
strainer = Strainer.new(context) strainer = Strainer.new(context)
@@filters.each { |k,m| strainer.extend(m) } @@filters.each { |m| strainer.extend(m) }
strainer strainer
end end

View File

@@ -23,4 +23,4 @@ module Liquid
end # Tag end # Tag
end # Tag end # Liquid

View File

@@ -10,24 +10,26 @@ module Liquid
# #
class Assign < Tag class Assign < Tag
Syntax = /(#{VariableSignature}+)\s*=\s*(.*)\s*/o Syntax = /(#{VariableSignature}+)\s*=\s*(.*)\s*/o
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
if markup =~ Syntax if markup =~ Syntax
@to = $1 @to = $1
@from = Variable.new($2) @from = Variable.new($2)
else else
raise SyntaxError.new("Syntax Error in 'assign' - Valid syntax: assign [var] = [source]") raise SyntaxError.new("Syntax Error in 'assign' - Valid syntax: assign [var] = [source]")
end end
super super
end end
def render(context) def render(context)
context.scopes.last[@to] = @from.render(context) val = @from.render(context)
'' context.scopes.last[@to] = val
end context.resource_limits[:assign_score_current] += (val.respond_to?(:length) ? val.length : 1)
''
end end
Template.register_tag('assign', Assign) end
Template.register_tag('assign', Assign)
end end

View File

@@ -8,8 +8,8 @@ module Liquid
# {% break %} # {% break %}
# {% endif %} # {% endif %}
# {% endfor %} # {% endfor %}
# #
class Break < Tag class Break < Tag
def interrupt def interrupt
BreakInterrupt.new BreakInterrupt.new

View File

@@ -27,6 +27,7 @@ module Liquid
def render(context) def render(context)
output = super output = super
context.scopes.last[@to] = output context.scopes.last[@to] = output
context.resource_limits[:assign_score_current] += (output.respond_to?(:length) ? output.length : 1)
'' ''
end end
end end

View File

@@ -3,15 +3,15 @@ module Liquid
Syntax = /(#{QuotedFragment})/o Syntax = /(#{QuotedFragment})/o
WhenSyntax = /(#{QuotedFragment})(?:(?:\s+or\s+|\s*\,\s*)(#{QuotedFragment}.*))?/o WhenSyntax = /(#{QuotedFragment})(?:(?:\s+or\s+|\s*\,\s*)(#{QuotedFragment}.*))?/o
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
@blocks = [] @blocks = []
if markup =~ Syntax if markup =~ Syntax
@left = $1 @left = $1
else else
raise SyntaxError.new("Syntax Error in tag 'case' - Valid syntax: case [condition]") raise SyntaxError.new("Syntax Error in tag 'case' - Valid syntax: case [condition]")
end end
super super
end end
@@ -27,53 +27,53 @@ module Liquid
end end
end end
def render(context) def render(context)
context.stack do context.stack do
execute_else_block = true execute_else_block = true
output = '' output = ''
@blocks.each do |block| @blocks.each do |block|
if block.else? if block.else?
return render_all(block.attachment, context) if execute_else_block return render_all(block.attachment, context) if execute_else_block
elsif block.evaluate(context) elsif block.evaluate(context)
execute_else_block = false execute_else_block = false
output << render_all(block.attachment, context) output << render_all(block.attachment, context)
end end
end end
output output
end
end
private
def record_when_condition(markup)
while markup
# Create a new nodelist and assign it to the new block
if not markup =~ WhenSyntax
raise SyntaxError.new("Syntax Error in tag 'case' - Valid when condition: {% when [condition] [or condition2...] %} ")
end
markup = $2
block = Condition.new(@left, '==', $1)
block.attach(@nodelist)
@blocks.push(block)
end end
end end
def record_else_condition(markup) private
def record_when_condition(markup)
while markup
# Create a new nodelist and assign it to the new block
if not markup =~ WhenSyntax
raise SyntaxError.new("Syntax Error in tag 'case' - Valid when condition: {% when [condition] [or condition2...] %} ")
end
markup = $2
block = Condition.new(@left, '==', $1)
block.attach(@nodelist)
@blocks.push(block)
end
end
def record_else_condition(markup)
if not markup.strip.empty? if not markup.strip.empty?
raise SyntaxError.new("Syntax Error in tag 'case' - Valid else condition: {% else %} (no parameters) ") raise SyntaxError.new("Syntax Error in tag 'case' - Valid else condition: {% else %} (no parameters) ")
end end
block = ElseCondition.new block = ElseCondition.new
block.attach(@nodelist) block.attach(@nodelist)
@blocks << block @blocks << block
end end
end end
Template.register_tag('case', Case) Template.register_tag('case', Case)
end end

View File

@@ -8,7 +8,7 @@ module Liquid
# {% continue %} # {% continue %}
# {% endif %} # {% endif %}
# {% endfor %} # {% endfor %}
# #
class Continue < Tag class Continue < Tag
def interrupt def interrupt

View File

@@ -1,5 +1,5 @@
module Liquid module Liquid
# Cycle is usually used within a loop to alternate between values, like colors or DOM classes. # Cycle is usually used within a loop to alternate between values, like colors or DOM classes.
# #
# {% for item in items %} # {% for item in items %}
@@ -15,45 +15,45 @@ module Liquid
class Cycle < Tag class Cycle < Tag
SimpleSyntax = /^#{QuotedFragment}+/o SimpleSyntax = /^#{QuotedFragment}+/o
NamedSyntax = /^(#{QuotedFragment})\s*\:\s*(.*)/o NamedSyntax = /^(#{QuotedFragment})\s*\:\s*(.*)/o
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
case markup case markup
when NamedSyntax when NamedSyntax
@variables = variables_from_string($2) @variables = variables_from_string($2)
@name = $1 @name = $1
when SimpleSyntax when SimpleSyntax
@variables = variables_from_string(markup) @variables = variables_from_string(markup)
@name = "'#{@variables.to_s}'" @name = "'#{@variables.to_s}'"
else else
raise SyntaxError.new("Syntax Error in 'cycle' - Valid syntax: cycle [name :] var [, var2, var3 ...]") raise SyntaxError.new("Syntax Error in 'cycle' - Valid syntax: cycle [name :] var [, var2, var3 ...]")
end end
super super
end end
def render(context) def render(context)
context.registers[:cycle] ||= Hash.new(0) context.registers[:cycle] ||= Hash.new(0)
context.stack do context.stack do
key = context[@name] key = context[@name]
iteration = context.registers[:cycle][key] iteration = context.registers[:cycle][key]
result = context[@variables[iteration]] result = context[@variables[iteration]]
iteration += 1 iteration += 1
iteration = 0 if iteration >= @variables.size iteration = 0 if iteration >= @variables.size
context.registers[:cycle][key] = iteration context.registers[:cycle][key] = iteration
result result
end end
end end
private private
def variables_from_string(markup) def variables_from_string(markup)
markup.split(',').collect do |var| markup.split(',').collect do |var|
var =~ /\s*(#{QuotedFragment})\s*/o var =~ /\s*(#{QuotedFragment})\s*/o
$1 ? $1 : nil $1 ? $1 : nil
end.compact end.compact
end end
end end
Template.register_tag('cycle', Cycle) Template.register_tag('cycle', Cycle)
end end

View File

@@ -1,5 +1,5 @@
module Liquid module Liquid
# decrement is used in a place where one needs to insert a counter # decrement is used in a place where one needs to insert a counter
# into a template, and needs the counter to survive across # into a template, and needs the counter to survive across
# multiple instantiations of the template. # multiple instantiations of the template.
@@ -19,21 +19,21 @@ module Liquid
# Hello: -3 # Hello: -3
# #
class Decrement < Tag class Decrement < Tag
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
@variable = markup.strip @variable = markup.strip
super super
end end
def render(context) def render(context)
value = context.environments.first[@variable] ||= 0 value = context.environments.first[@variable] ||= 0
value = value - 1 value = value - 1
context.environments.first[@variable] = value context.environments.first[@variable] = value
value.to_s value.to_s
end end
private private
end end
Template.register_tag('decrement', Decrement) Template.register_tag('decrement', Decrement)
end end

View File

@@ -1,6 +1,6 @@
module Liquid module Liquid
# "For" iterates over an array or collection. # "For" iterates over an array or collection.
# Several useful variables are available to you within the loop. # Several useful variables are available to you within the loop.
# #
# == Basic usage: # == Basic usage:
@@ -22,7 +22,7 @@ module Liquid
# #
# {% for item in collection limit:5 offset:10 %} # {% for item in collection limit:5 offset:10 %}
# {{ item.name }} # {{ item.name }}
# {% end %} # {% end %}
# #
# To reverse the for loop simply use {% for item in collection reversed %} # To reverse the for loop simply use {% for item in collection reversed %}
# #
@@ -31,7 +31,7 @@ module Liquid
# forloop.name:: 'item-collection' # forloop.name:: 'item-collection'
# forloop.length:: Length of the loop # forloop.length:: Length of the loop
# forloop.index:: The current item's position in the collection; # forloop.index:: The current item's position in the collection;
# forloop.index starts at 1. # forloop.index starts at 1.
# This is helpful for non-programmers who start believe # This is helpful for non-programmers who start believe
# the first item in an array is 1, not 0. # the first item in an array is 1, not 0.
# forloop.index0:: The current item's position in the collection # forloop.index0:: The current item's position in the collection
@@ -43,19 +43,19 @@ module Liquid
# forloop.first:: Returns true if the item is the first item. # forloop.first:: Returns true if the item is the first item.
# forloop.last:: Returns true if the item is the last item. # forloop.last:: Returns true if the item is the last item.
# #
class For < Block class For < Block
Syntax = /(\w+)\s+in\s+(#{QuotedFragment}+)\s*(reversed)?/o Syntax = /\A(\w+)\s+in\s+(#{QuotedFragment}+)\s*(reversed)?/o
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
if markup =~ Syntax if markup =~ Syntax
@variable_name = $1 @variable_name = $1
@collection_name = $2 @collection_name = $2
@name = "#{$1}-#{$2}" @name = "#{$1}-#{$2}"
@reversed = $3 @reversed = $3
@attributes = {} @attributes = {}
markup.scan(TagAttributes) do |key, value| markup.scan(TagAttributes) do |key, value|
@attributes[key] = value @attributes[key] = value
end end
else else
raise SyntaxError.new("Syntax Error in 'for loop' - Valid syntax: for [item] in [collection]") raise SyntaxError.new("Syntax Error in 'for loop' - Valid syntax: for [item] in [collection]")
end end
@@ -68,51 +68,51 @@ module Liquid
return super unless tag == 'else' return super unless tag == 'else'
@nodelist = @else_block = [] @nodelist = @else_block = []
end end
def render(context) def render(context)
context.registers[:for] ||= Hash.new(0) context.registers[:for] ||= Hash.new(0)
collection = context[@collection_name] collection = context[@collection_name]
collection = collection.to_a if collection.is_a?(Range) collection = collection.to_a if collection.is_a?(Range)
# Maintains Ruby 1.8.7 String#each behaviour on 1.9 # Maintains Ruby 1.8.7 String#each behaviour on 1.9
return render_else(context) unless iterable?(collection) return render_else(context) unless iterable?(collection)
from = if @attributes['offset'] == 'continue' from = if @attributes['offset'] == 'continue'
context.registers[:for][@name].to_i context.registers[:for][@name].to_i
else else
context[@attributes['offset']].to_i context[@attributes['offset']].to_i
end end
limit = context[@attributes['limit']] limit = context[@attributes['limit']]
to = limit ? limit.to_i + from : nil to = limit ? limit.to_i + from : nil
segment = Utils.slice_collection_using_each(collection, from, to) segment = Utils.slice_collection_using_each(collection, from, to)
return render_else(context) if segment.empty? return render_else(context) if segment.empty?
segment.reverse! if @reversed segment.reverse! if @reversed
result = '' result = ''
length = segment.length length = segment.length
# Store our progress through the collection for the continue flag # Store our progress through the collection for the continue flag
context.registers[:for][@name] = from + segment.length context.registers[:for][@name] = from + segment.length
context.stack do context.stack do
segment.each_with_index do |item, index| segment.each_with_index do |item, index|
context[@variable_name] = item context[@variable_name] = item
context['forloop'] = { context['forloop'] = {
'name' => @name, 'name' => @name,
'length' => length, 'length' => length,
'index' => index + 1, 'index' => index + 1,
'index0' => index, 'index0' => index,
'rindex' => length - index, 'rindex' => length - index,
'rindex0' => length - index - 1, 'rindex0' => length - index - 1,
'first' => (index == 0), 'first' => (index == 0),
'last' => (index == length - 1) } 'last' => (index == length - 1) }
result << render_all(@for_block, context) result << render_all(@for_block, context)
@@ -124,8 +124,8 @@ module Liquid
end end
end end
end end
result result
end end
private private

View File

@@ -1,20 +1,20 @@
module Liquid module Liquid
class Ifchanged < Block class Ifchanged < Block
def render(context) def render(context)
context.stack do context.stack do
output = render_all(@nodelist, context) output = render_all(@nodelist, context)
if output != context.registers[:ifchanged] if output != context.registers[:ifchanged]
context.registers[:ifchanged] = output context.registers[:ifchanged] = output
output output
else else
'' ''
end end
end end
end end
end end
Template.register_tag('ifchanged', Ifchanged) Template.register_tag('ifchanged', Ifchanged)
end end

View File

@@ -1,11 +1,26 @@
module Liquid module Liquid
# Include allows templates to relate with other templates
#
# Simply include another template:
#
# {% include 'product' %}
#
# Include a template with a local variable:
#
# {% include 'product' with products[0] %}
#
# Include a template for a collection:
#
# {% include 'product' for products %}
#
class Include < Tag class Include < Tag
Syntax = /(#{QuotedFragment}+)(\s+(?:with|for)\s+(#{QuotedFragment}+))?/o Syntax = /(#{QuotedFragment}+)(\s+(?:with|for)\s+(#{QuotedFragment}+))?/o
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
if markup =~ Syntax if markup =~ Syntax
@template_name = $1 @template_name = $1
@variable_name = $3 @variable_name = $3
@attributes = {} @attributes = {}
@@ -19,23 +34,22 @@ module Liquid
super super
end end
def parse(tokens) def parse(tokens)
end end
def render(context) def render(context)
source = _read_template_from_file_system(context) partial = load_cached_partial(context)
partial = Liquid::Template.parse(source)
variable = context[@variable_name || @template_name[1..-2]] variable = context[@variable_name || @template_name[1..-2]]
context.stack do context.stack do
@attributes.each do |key, value| @attributes.each do |key, value|
context[key] = context[value] context[key] = context[value]
end end
if variable.is_a?(Array) if variable.is_a?(Array)
variable.collect do |variable| variable.collect do |var|
context[@template_name[1..-2]] = variable context[@template_name[1..-2]] = var
partial.render(context) partial.render(context)
end end
else else
@@ -44,11 +58,25 @@ module Liquid
end end
end end
end end
private private
def _read_template_from_file_system(context) def load_cached_partial(context)
cached_partials = context.registers[:cached_partials] || {}
template_name = context[@template_name]
if cached = cached_partials[template_name]
return cached
end
source = read_template_from_file_system(context)
partial = Liquid::Template.parse(source)
cached_partials[template_name] = partial
context.registers[:cached_partials] = cached_partials
partial
end
def read_template_from_file_system(context)
file_system = context.registers[:file_system] || Liquid::Template.file_system file_system = context.registers[:file_system] || Liquid::Template.file_system
# make read_template_file call backwards-compatible. # make read_template_file call backwards-compatible.
case file_system.method(:read_template_file).arity case file_system.method(:read_template_file).arity
when 1 when 1
@@ -61,5 +89,5 @@ module Liquid
end end
end end
Template.register_tag('include', Include) Template.register_tag('include', Include)
end end

View File

@@ -1,5 +1,5 @@
module Liquid module Liquid
# increment is used in a place where one needs to insert a counter # increment is used in a place where one needs to insert a counter
# into a template, and needs the counter to survive across # into a template, and needs the counter to survive across
# multiple instantiations of the template. # multiple instantiations of the template.
@@ -16,20 +16,20 @@ module Liquid
# Hello: 2 # Hello: 2
# #
class Increment < Tag class Increment < Tag
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
@variable = markup.strip @variable = markup.strip
super super
end end
def render(context) def render(context)
value = context.environments.first[@variable] ||= 0 value = context.environments.first[@variable] ||= 0
context.environments.first[@variable] = value + 1 context.environments.first[@variable] = value + 1
value.to_s value.to_s
end end
private private
end end
Template.register_tag('increment', Increment) Template.register_tag('increment', Increment)
end end

View File

@@ -1,12 +1,14 @@
module Liquid module Liquid
class Raw < Block class Raw < Block
FullTokenPossiblyInvalid = /^(.*)#{TagStart}\s*(\w+)\s*(.*)?#{TagEnd}$/o
def parse(tokens) def parse(tokens)
@nodelist ||= [] @nodelist ||= []
@nodelist.clear @nodelist.clear
while token = tokens.shift while token = tokens.shift
if token =~ FullToken if token =~ FullTokenPossiblyInvalid
if block_delimiter == $1 @nodelist << $1 if $1 != ""
if block_delimiter == $2
end_tag end_tag
return return
end end
@@ -18,4 +20,3 @@ module Liquid
Template.register_tag('raw', Raw) Template.register_tag('raw', Raw)
end end

View File

@@ -9,25 +9,25 @@ module Liquid
class Unless < If class Unless < If
def render(context) def render(context)
context.stack do context.stack do
# First condition is interpreted backwards ( if not ) # First condition is interpreted backwards ( if not )
block = @blocks.first first_block = @blocks.first
unless block.evaluate(context) unless first_block.evaluate(context)
return render_all(block.attachment, context) return render_all(first_block.attachment, context)
end end
# After the first condition unless works just like if # After the first condition unless works just like if
@blocks[1..-1].each do |block| @blocks[1..-1].each do |block|
if block.evaluate(context) if block.evaluate(context)
return render_all(block.attachment, context) return render_all(block.attachment, context)
end end
end end
'' ''
end end
end end
end end
Template.register_tag('unless', Unless) Template.register_tag('unless', Unless)
end end

View File

@@ -14,7 +14,7 @@ module Liquid
# template.render('user_name' => 'bob') # template.render('user_name' => 'bob')
# #
class Template class Template
attr_accessor :root attr_accessor :root, :resource_limits
@@file_system = BlankFileSystem.new @@file_system = BlankFileSystem.new
class << self class << self
@@ -50,6 +50,7 @@ module Liquid
# creates a new <tt>Template</tt> from an array of tokens. Use <tt>Template.parse</tt> instead # creates a new <tt>Template</tt> from an array of tokens. Use <tt>Template.parse</tt> instead
def initialize def initialize
@resource_limits = {}
end end
# Parse source code. # Parse source code.
@@ -88,14 +89,17 @@ module Liquid
# #
def render(*args) def render(*args)
return '' if @root.nil? return '' if @root.nil?
context = case args.first context = case args.first
when Liquid::Context when Liquid::Context
args.shift args.shift
when Liquid::Drop
drop = args.shift
drop.context = Context.new([drop, assigns], instance_assigns, registers, @rethrow_errors, @resource_limits)
when Hash when Hash
Context.new([args.shift, assigns], instance_assigns, registers, @rethrow_errors) Context.new([args.shift, assigns], instance_assigns, registers, @rethrow_errors, @resource_limits)
when nil when nil
Context.new(assigns, instance_assigns, registers, @rethrow_errors) Context.new(assigns, instance_assigns, registers, @rethrow_errors, @resource_limits)
else else
raise ArgumentError, "Expect Hash or Liquid::Context as parameter" raise ArgumentError, "Expect Hash or Liquid::Context as parameter"
end end
@@ -120,9 +124,11 @@ module Liquid
begin begin
# render the nodelist. # render the nodelist.
# for performance reasons we get a array back here. join will make a string out of it # for performance reasons we get an array back here. join will make a string out of it.
result = @root.render(context) result = @root.render(context)
result.respond_to?(:join) ? result.join : result result.respond_to?(:join) ? result.join : result
rescue Liquid::MemoryError => e
context.handle_error(e)
ensure ensure
@errors = context.errors @errors = context.errors
end end

View File

@@ -3,7 +3,6 @@ module Liquid
def self.slice_collection_using_each(collection, from, to) def self.slice_collection_using_each(collection, from, to)
segments = [] segments = []
index = 0 index = 0
yielded = 0
# Maintains Ruby 1.8.7 String#each behaviour on 1.9 # Maintains Ruby 1.8.7 String#each behaviour on 1.9
return [collection] if non_blank_string?(collection) return [collection] if non_blank_string?(collection)

View File

@@ -23,9 +23,9 @@ module Liquid
if match[2].match(/#{FilterSeparator}\s*(.*)/o) if match[2].match(/#{FilterSeparator}\s*(.*)/o)
filters = Regexp.last_match(1).scan(FilterParser) filters = Regexp.last_match(1).scan(FilterParser)
filters.each do |f| filters.each do |f|
if matches = f.match(/\s*(\w+)(?:\s*#{FilterArgumentSeparator}(.*))?/) if matches = f.match(/\s*(\w+)/)
filtername = matches[1] filtername = matches[1]
filterargs = matches[2].to_s.scan(/(?:\A|#{ArgumentSeparator})\s*((?:\w+\s*\:\s*)?#{QuotedFragment})/o).flatten filterargs = f.scan(/(?:#{FilterArgumentSeparator}|#{ArgumentSeparator})\s*((?:\w+\s*\:\s*)?#{QuotedFragment})/o).flatten
@filters << [filtername, filterargs] @filters << [filtername, filterargs]
end end
end end

4
lib/liquid/version.rb Normal file
View File

@@ -0,0 +1,4 @@
# encoding: utf-8
module Liquid
VERSION = "2.6.3"
end

View File

@@ -1,8 +1,12 @@
# encoding: utf-8 # encoding: utf-8
lib = File.expand_path('../lib/', __FILE__)
$:.unshift lib unless $:.include?(lib)
require "liquid/version"
Gem::Specification.new do |s| Gem::Specification.new do |s|
s.name = "liquid" s.name = "liquid"
s.version = "2.5.5" s.version = Liquid::VERSION
s.platform = Gem::Platform::RUBY s.platform = Gem::Platform::RUBY
s.summary = "A secure, non-evaling end user template engine with aesthetic markup." s.summary = "A secure, non-evaling end user template engine with aesthetic markup."
s.authors = ["Tobias Luetke"] s.authors = ["Tobias Luetke"]

View File

@@ -1,5 +1,5 @@
class CommentForm < Liquid::Block class CommentForm < Liquid::Block
Syntax = /(#{Liquid::VariableSignature}+)/ Syntax = /(#{Liquid::VariableSignature}+)/
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
if markup =~ Syntax if markup =~ Syntax
@@ -8,14 +8,14 @@ class CommentForm < Liquid::Block
else else
raise SyntaxError.new("Syntax Error in 'comment_form' - Valid syntax: comment_form [article]") raise SyntaxError.new("Syntax Error in 'comment_form' - Valid syntax: comment_form [article]")
end end
super super
end end
def render(context) def render(context)
article = context[@variable_name] article = context[@variable_name]
context.stack do context.stack do
context['form'] = { context['form'] = {
'posted_successfully?' => context.registers[:posted_successfully], 'posted_successfully?' => context.registers[:posted_successfully],
'errors' => context['comment.errors'], 'errors' => context['comment.errors'],
@@ -25,9 +25,9 @@ class CommentForm < Liquid::Block
} }
wrap_in_form(article, render_all(@nodelist, context)) wrap_in_form(article, render_all(@nodelist, context))
end end
end end
def wrap_in_form(article, input) def wrap_in_form(article, input)
%Q{<form id="article-#{article.id}-comment-form" class="comment-form" method="post" action="">\n#{input}\n</form>} %Q{<form id="article-#{article.id}-comment-form" class="comment-form" method="post" action="">\n#{input}\n</form>}
end end
end end

View File

@@ -1,45 +1,45 @@
require 'yaml' require 'yaml'
module Database module Database
# Load the standard vision toolkit database and re-arrage it to be simply exportable # Load the standard vision toolkit database and re-arrage it to be simply exportable
# to liquid as assigns. All this is based on Shopify # to liquid as assigns. All this is based on Shopify
def self.tables def self.tables
@tables ||= begin @tables ||= begin
db = YAML.load_file(File.dirname(__FILE__) + '/vision.database.yml') db = YAML.load_file(File.dirname(__FILE__) + '/vision.database.yml')
# From vision source # From vision source
db['products'].each do |product| db['products'].each do |product|
collections = db['collections'].find_all do |collection| collections = db['collections'].find_all do |collection|
collection['products'].any? { |p| p['id'].to_i == product['id'].to_i } collection['products'].any? { |p| p['id'].to_i == product['id'].to_i }
end end
product['collections'] = collections product['collections'] = collections
end end
# key the tables by handles, as this is how liquid expects it. # key the tables by handles, as this is how liquid expects it.
db = db.inject({}) do |assigns, (key, values)| db = db.inject({}) do |assigns, (key, values)|
assigns[key] = values.inject({}) { |h, v| h[v['handle']] = v; h; } assigns[key] = values.inject({}) { |h, v| h[v['handle']] = v; h; }
assigns assigns
end end
# Some standard direct accessors so that the specialized templates # Some standard direct accessors so that the specialized templates
# render correctly # render correctly
db['collection'] = db['collections'].values.first db['collection'] = db['collections'].values.first
db['product'] = db['products'].values.first db['product'] = db['products'].values.first
db['blog'] = db['blogs'].values.first db['blog'] = db['blogs'].values.first
db['article'] = db['blog']['articles'].first db['article'] = db['blog']['articles'].first
db['cart'] = { db['cart'] = {
'total_price' => db['line_items'].values.inject(0) { |sum, item| sum += item['line_price'] * item['quantity'] }, 'total_price' => db['line_items'].values.inject(0) { |sum, item| sum += item['line_price'] * item['quantity'] },
'item_count' => db['line_items'].values.inject(0) { |sum, item| sum += item['quantity'] }, 'item_count' => db['line_items'].values.inject(0) { |sum, item| sum += item['quantity'] },
'items' => db['line_items'].values 'items' => db['line_items'].values
} }
db db
end end
end end
end end
if __FILE__ == $0 if __FILE__ == $0
p Database.tables['collections']['frontpage'].keys p Database.tables['collections']['frontpage'].keys
#p Database.tables['blog']['articles'] #p Database.tables['blog']['articles']
end end

View File

@@ -1,7 +1,7 @@
module JsonFilter module JsonFilter
def json(object) def json(object)
object.reject {|k,v| k == "collections" }.to_json object.reject {|k,v| k == "collections" }.to_json
end end
end end

View File

@@ -1,5 +1,5 @@
module MoneyFilter module MoneyFilter
def money_with_currency(money) def money_with_currency(money)
return '' if money.nil? return '' if money.nil?
sprintf("$ %.2f USD", money/100.0) sprintf("$ %.2f USD", money/100.0)
@@ -9,10 +9,10 @@ module MoneyFilter
return '' if money.nil? return '' if money.nil?
sprintf("$ %.2f", money/100.0) sprintf("$ %.2f", money/100.0)
end end
private private
def currency def currency
ShopDrop.new.currency ShopDrop.new.currency
end end
end end

View File

@@ -3,7 +3,7 @@ class Paginate < Liquid::Block
def initialize(tag_name, markup, tokens) def initialize(tag_name, markup, tokens)
@nodelist = [] @nodelist = []
if markup =~ Syntax if markup =~ Syntax
@collection_name = $1 @collection_name = $1
@page_size = if $2 @page_size = if $2
@@ -11,22 +11,22 @@ class Paginate < Liquid::Block
else else
20 20
end end
@attributes = { 'window_size' => 3 } @attributes = { 'window_size' => 3 }
markup.scan(Liquid::TagAttributes) do |key, value| markup.scan(Liquid::TagAttributes) do |key, value|
@attributes[key] = value @attributes[key] = value
end end
else else
raise SyntaxError.new("Syntax Error in tag 'paginate' - Valid syntax: paginate [collection] by number") raise SyntaxError.new("Syntax Error in tag 'paginate' - Valid syntax: paginate [collection] by number")
end end
super super
end end
def render(context) def render(context)
@context = context @context = context
context.stack do context.stack do
current_page = context['current_page'].to_i current_page = context['current_page'].to_i
pagination = { pagination = {
@@ -34,29 +34,29 @@ class Paginate < Liquid::Block
'current_page' => 5, 'current_page' => 5,
'current_offset' => @page_size * 5 'current_offset' => @page_size * 5
} }
context['paginate'] = pagination context['paginate'] = pagination
collection_size = context[@collection_name].size collection_size = context[@collection_name].size
raise ArgumentError.new("Cannot paginate array '#{@collection_name}'. Not found.") if collection_size.nil? raise ArgumentError.new("Cannot paginate array '#{@collection_name}'. Not found.") if collection_size.nil?
page_count = (collection_size.to_f / @page_size.to_f).to_f.ceil + 1 page_count = (collection_size.to_f / @page_size.to_f).to_f.ceil + 1
pagination['items'] = collection_size pagination['items'] = collection_size
pagination['pages'] = page_count -1 pagination['pages'] = page_count -1
pagination['previous'] = link('&laquo; Previous', current_page-1 ) unless 1 >= current_page pagination['previous'] = link('&laquo; Previous', current_page-1 ) unless 1 >= current_page
pagination['next'] = link('Next &raquo;', current_page+1 ) unless page_count <= current_page+1 pagination['next'] = link('Next &raquo;', current_page+1 ) unless page_count <= current_page+1
pagination['parts'] = [] pagination['parts'] = []
hellip_break = false hellip_break = false
if page_count > 2 if page_count > 2
1.upto(page_count-1) do |page| 1.upto(page_count-1) do |page|
if current_page == page if current_page == page
pagination['parts'] << no_link(page) pagination['parts'] << no_link(page)
elsif page == 1 elsif page == 1
pagination['parts'] << link(page, page) pagination['parts'] << link(page, page)
elsif page == page_count -1 elsif page == page_count -1
pagination['parts'] << link(page, page) pagination['parts'] << link(page, page)
@@ -68,26 +68,26 @@ class Paginate < Liquid::Block
else else
pagination['parts'] << link(page, page) pagination['parts'] << link(page, page)
end end
hellip_break = false hellip_break = false
end end
end end
render_all(@nodelist, context) render_all(@nodelist, context)
end end
end end
private private
def no_link(title) def no_link(title)
{ 'title' => title, 'is_link' => false} { 'title' => title, 'is_link' => false}
end end
def link(title, page) def link(title, page)
{ 'title' => title, 'url' => current_url + "?page=#{page}", 'is_link' => true} { 'title' => title, 'url' => current_url + "?page=#{page}", 'is_link' => true}
end end
def current_url def current_url
"/collections/frontpage" "/collections/frontpage"
end end
end end

View File

@@ -1,5 +1,5 @@
module ShopFilter module ShopFilter
def asset_url(input) def asset_url(input)
"/files/1/[shop_id]/[shop_id]/assets/#{input}" "/files/1/[shop_id]/[shop_id]/assets/#{input}"
end end
@@ -7,11 +7,11 @@ module ShopFilter
def global_asset_url(input) def global_asset_url(input)
"/global/#{input}" "/global/#{input}"
end end
def shopify_asset_url(input) def shopify_asset_url(input)
"/shopify/#{input}" "/shopify/#{input}"
end end
def script_tag(url) def script_tag(url)
%(<script src="#{url}" type="text/javascript"></script>) %(<script src="#{url}" type="text/javascript"></script>)
end end
@@ -19,15 +19,15 @@ module ShopFilter
def stylesheet_tag(url, media="all") def stylesheet_tag(url, media="all")
%(<link href="#{url}" rel="stylesheet" type="text/css" media="#{media}" />) %(<link href="#{url}" rel="stylesheet" type="text/css" media="#{media}" />)
end end
def link_to(link, url, title="") def link_to(link, url, title="")
%|<a href="#{url}" title="#{title}">#{link}</a>| %|<a href="#{url}" title="#{title}">#{link}</a>|
end end
def img_tag(url, alt="") def img_tag(url, alt="")
%|<img src="#{url}" alt="#{alt}" />| %|<img src="#{url}" alt="#{alt}" />|
end end
def link_to_vendor(vendor) def link_to_vendor(vendor)
if vendor if vendor
link_to vendor, url_for_vendor(vendor), vendor link_to vendor, url_for_vendor(vendor), vendor
@@ -35,7 +35,7 @@ module ShopFilter
'Unknown Vendor' 'Unknown Vendor'
end end
end end
def link_to_type(type) def link_to_type(type)
if type if type
link_to type, url_for_type(type), type link_to type, url_for_type(type), type
@@ -43,56 +43,56 @@ module ShopFilter
'Unknown Vendor' 'Unknown Vendor'
end end
end end
def url_for_vendor(vendor_title) def url_for_vendor(vendor_title)
"/collections/#{vendor_title.to_handle}" "/collections/#{vendor_title.to_handle}"
end end
def url_for_type(type_title) def url_for_type(type_title)
"/collections/#{type_title.to_handle}" "/collections/#{type_title.to_handle}"
end end
def product_img_url(url, style = 'small') def product_img_url(url, style = 'small')
unless url =~ /^products\/([\w\-\_]+)\.(\w{2,4})/ unless url =~ /^products\/([\w\-\_]+)\.(\w{2,4})/
raise ArgumentError, 'filter "size" can only be called on product images' raise ArgumentError, 'filter "size" can only be called on product images'
end end
case style case style
when 'original' when 'original'
return '/files/shops/random_number/' + url return '/files/shops/random_number/' + url
when 'grande', 'large', 'medium', 'compact', 'small', 'thumb', 'icon' when 'grande', 'large', 'medium', 'compact', 'small', 'thumb', 'icon'
"/files/shops/random_number/products/#{$1}_#{style}.#{$2}" "/files/shops/random_number/products/#{$1}_#{style}.#{$2}"
else else
raise ArgumentError, 'valid parameters for filter "size" are: original, grande, large, medium, compact, small, thumb and icon ' raise ArgumentError, 'valid parameters for filter "size" are: original, grande, large, medium, compact, small, thumb and icon '
end end
end end
def default_pagination(paginate) def default_pagination(paginate)
html = [] html = []
html << %(<span class="prev">#{link_to(paginate['previous']['title'], paginate['previous']['url'])}</span>) if paginate['previous'] html << %(<span class="prev">#{link_to(paginate['previous']['title'], paginate['previous']['url'])}</span>) if paginate['previous']
for part in paginate['parts'] for part in paginate['parts']
if part['is_link'] if part['is_link']
html << %(<span class="page">#{link_to(part['title'], part['url'])}</span>) html << %(<span class="page">#{link_to(part['title'], part['url'])}</span>)
elsif part['title'].to_i == paginate['current_page'].to_i elsif part['title'].to_i == paginate['current_page'].to_i
html << %(<span class="page current">#{part['title']}</span>) html << %(<span class="page current">#{part['title']}</span>)
else else
html << %(<span class="deco">#{part['title']}</span>) html << %(<span class="deco">#{part['title']}</span>)
end end
end end
html << %(<span class="next">#{link_to(paginate['next']['title'], paginate['next']['url'])}</span>) if paginate['next'] html << %(<span class="next">#{link_to(paginate['next']['title'], paginate['next']['url'])}</span>) if paginate['next']
html.join(' ') html.join(' ')
end end
# Accepts a number, and two words - one for singular, one for plural # Accepts a number, and two words - one for singular, one for plural
# Returns the singular word if input equals 1, otherwise plural # Returns the singular word if input equals 1, otherwise plural
def pluralize(input, singular, plural) def pluralize(input, singular, plural)
input == 1 ? singular : plural input == 1 ? singular : plural
end end
end end

View File

@@ -1,9 +1,9 @@
module TagFilter module TagFilter
def link_to_tag(label, tag) def link_to_tag(label, tag)
"<a title=\"Show tag #{tag}\" href=\"/collections/#{@context['handle']}/#{tag}\">#{label}</a>" "<a title=\"Show tag #{tag}\" href=\"/collections/#{@context['handle']}/#{tag}\">#{label}</a>"
end end
def highlight_active_tag(tag, css_class='active') def highlight_active_tag(tag, css_class='active')
if @context['current_tags'].include?(tag) if @context['current_tags'].include?(tag)
"<span class=\"#{css_class}\">#{tag}</span>" "<span class=\"#{css_class}\">#{tag}</span>"
@@ -12,14 +12,14 @@ module TagFilter
end end
end end
def link_to_add_tag(label, tag) def link_to_add_tag(label, tag)
tags = (@context['current_tags'] + [tag]).uniq tags = (@context['current_tags'] + [tag]).uniq
"<a title=\"Show tag #{tag}\" href=\"/collections/#{@context['handle']}/#{tags.join("+")}\">#{label}</a>" "<a title=\"Show tag #{tag}\" href=\"/collections/#{@context['handle']}/#{tags.join("+")}\">#{label}</a>"
end end
def link_to_remove_tag(label, tag) def link_to_remove_tag(label, tag)
tags = (@context['current_tags'] - [tag]).uniq tags = (@context['current_tags'] - [tag]).uniq
"<a title=\"Show tag #{tag}\" href=\"/collections/#{@context['handle']}/#{tags.join("+")}\">#{label}</a>" "<a title=\"Show tag #{tag}\" href=\"/collections/#{@context['handle']}/#{tags.join("+")}\">#{label}</a>"
end end
end end

View File

@@ -2,7 +2,7 @@
# Variants # Variants
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
product_variants: product_variants:
- &product-1-var-1 - &product-1-var-1
id: 1 id: 1
title: 151cm / Normal title: 151cm / Normal
@@ -13,7 +13,7 @@ product_variants:
inventory_quantity: 5 inventory_quantity: 5
option1: 151cm option1: 151cm
option2: Normal option2: Normal
option3: option3:
- &product-1-var-2 - &product-1-var-2
id: 2 id: 2
title: 155cm / Normal title: 155cm / Normal
@@ -24,7 +24,7 @@ product_variants:
inventory_quantity: 2 inventory_quantity: 2
option1: 155cm option1: 155cm
option2: Normal option2: Normal
option3: option3:
- &product-2-var-1 - &product-2-var-1
id: 3 id: 3
title: 162cm title: 162cm
@@ -34,19 +34,19 @@ product_variants:
available: true available: true
inventory_quantity: 3 inventory_quantity: 3
option1: 162cm option1: 162cm
option2: option2:
option3: option3:
- &product-3-var-1 - &product-3-var-1
id: 4 id: 4
title: 159cm title: 159cm
price: 19900 price: 19900
weight: 1000 weight: 1000
compare_at_price: compare_at_price:
available: true available: true
inventory_quantity: 4 inventory_quantity: 4
option1: 159cm option1: 159cm
option2: option2:
option3: option3:
- &product-4-var-1 - &product-4-var-1
id: 5 id: 5
title: 159cm title: 159cm
@@ -56,8 +56,8 @@ product_variants:
available: true available: true
inventory_quantity: 6 inventory_quantity: 6
option1: 159cm option1: 159cm
option2: option2:
option3: option3:
- &product-1-var-3 - &product-1-var-3
id: 6 id: 6
title: 158cm / Wide title: 158cm / Wide
@@ -74,12 +74,12 @@ product_variants:
title: 162cm title: 162cm
price: 19900 price: 19900
weight: 1000 weight: 1000
compare_at_price: compare_at_price:
available: false available: false
inventory_quantity: 0 inventory_quantity: 0
option1: 162cm option1: 162cm
option2: option2:
option3: option3:
- &product-3-var-3 - &product-3-var-3
id: 8 id: 8
title: 165cm title: 165cm
@@ -89,8 +89,8 @@ product_variants:
available: true available: true
inventory_quantity: 4 inventory_quantity: 4
option1: 165cm option1: 165cm
option2: option2:
option3: option3:
- &product-5-var-1 - &product-5-var-1
id: 9 id: 9
title: black / 42 title: black / 42
@@ -233,7 +233,7 @@ product_variants:
inventory_quantity: 0 inventory_quantity: 0
option1: brown option1: brown
option2: medium option2: medium
option3: option3:
- &product-8-var-3 - &product-8-var-3
id: 22 id: 22
title: brown / large title: brown / large
@@ -315,7 +315,7 @@ product_variants:
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
# Products # Products
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
products: products:
- &product-1 - &product-1
id: 1 id: 1
@@ -327,27 +327,27 @@ products:
price_max: 31900 price_max: 31900
price_min: 23900 price_min: 23900
price_varies: true price_varies: true
available: true available: true
tags: tags:
- season2005 - season2005
- pro - pro
- intermediate - intermediate
- wooden - wooden
- freestyle - freestyle
options: options:
- Length - Length
- Style - Style
compare_at_price: 49900 compare_at_price: 49900
compare_at_price_max: 50900 compare_at_price_max: 50900
compare_at_price_min: 49900 compare_at_price_min: 49900
compare_at_price_varies: true compare_at_price_varies: true
url: /products/arbor-draft url: /products/arbor-draft
featured_image: products/arbor_draft.jpg featured_image: products/arbor_draft.jpg
images: images:
- products/arbor_draft.jpg - products/arbor_draft.jpg
description: description:
The Arbor Draft snowboard wouldn't exist if Polynesians hadn't figured out how to surf hundreds of years ago. But the Draft does exist, and it's here to bring your urban and park riding to a new level. The board's freaky Tiki design pays homage to culture that inspired snowboarding. It's designed to spin with ease, land smoothly, lock hook-free onto rails, and take the abuse of a pavement pounding or twelve. The Draft will pop off kickers with authority and carve solidly across the pipe. The Draft features targeted Koa wood die cuts inlayed into the deck that enhance the flex pattern. Now bow down to riding's ancestors. The Arbor Draft snowboard wouldn't exist if Polynesians hadn't figured out how to surf hundreds of years ago. But the Draft does exist, and it's here to bring your urban and park riding to a new level. The board's freaky Tiki design pays homage to culture that inspired snowboarding. It's designed to spin with ease, land smoothly, lock hook-free onto rails, and take the abuse of a pavement pounding or twelve. The Draft will pop off kickers with authority and carve solidly across the pipe. The Draft features targeted Koa wood die cuts inlayed into the deck that enhance the flex pattern. Now bow down to riding's ancestors.
variants: variants:
- *product-1-var-1 - *product-1-var-1
- *product-1-var-2 - *product-1-var-2
- *product-1-var-3 - *product-1-var-3
@@ -361,14 +361,14 @@ products:
price_max: 29900 price_max: 29900
price_min: 29900 price_min: 29900
price_varies: false price_varies: false
available: true available: true
tags: tags:
- season2005 - season2005
- pro - pro
- wooden - wooden
- freestyle - freestyle
options: options:
- Length - Length
compare_at_price: 52900 compare_at_price: 52900
compare_at_price_max: 52900 compare_at_price_max: 52900
compare_at_price_min: 52900 compare_at_price_min: 52900
@@ -379,29 +379,29 @@ products:
- products/element58.jpg - products/element58.jpg
description: description:
The Element is a technically advanced all-mountain board for riders who readily transition from one terrain, snow condition, or riding style to another. Its balanced design provides the versatility needed for the true ride-it-all experience. The Element is exceedingly lively, freely initiates, and holds a tight edge at speed. Its structural real-wood topsheet is made with book-matched Koa. The Element is a technically advanced all-mountain board for riders who readily transition from one terrain, snow condition, or riding style to another. Its balanced design provides the versatility needed for the true ride-it-all experience. The Element is exceedingly lively, freely initiates, and holds a tight edge at speed. Its structural real-wood topsheet is made with book-matched Koa.
variants: variants:
- *product-2-var-1 - *product-2-var-1
- &product-3 - &product-3
id: 3 id: 3
title: Comic ~ Pastel title: Comic ~ Pastel
handle: comic-pastel handle: comic-pastel
type: Snowboards type: Snowboards
vendor: Technine vendor: Technine
price: 19900 price: 19900
price_max: 22900 price_max: 22900
price_min: 19900 price_min: 19900
tags: tags:
- season2006 - season2006
- beginner - beginner
- intermediate - intermediate
- freestyle - freestyle
- purple - purple
options: options:
- Length - Length
price_varies: true price_varies: true
available: true available: true
compare_at_price: compare_at_price:
compare_at_price_max: 0 compare_at_price_max: 0
compare_at_price_min: 0 compare_at_price_min: 0
compare_at_price_varies: false compare_at_price_varies: false
@@ -413,7 +413,7 @@ products:
- products/technine_detail.jpg - products/technine_detail.jpg
description: description:
2005 Technine Comic Series Description The Comic series was developed to be the ultimate progressive freestyle board in the Technine line. Dependable edge control and a perfect flex pattern for jumping in the park or out of bounds. Landins and progression will come easy with this board and it will help your riding progress to the next level. Street rails, park jibs, backcountry booters and park jumps, this board will do it all. 2005 Technine Comic Series Description The Comic series was developed to be the ultimate progressive freestyle board in the Technine line. Dependable edge control and a perfect flex pattern for jumping in the park or out of bounds. Landins and progression will come easy with this board and it will help your riding progress to the next level. Street rails, park jibs, backcountry booters and park jumps, this board will do it all.
variants: variants:
- *product-3-var-1 - *product-3-var-1
- *product-3-var-2 - *product-3-var-2
- *product-3-var-3 - *product-3-var-3
@@ -428,12 +428,12 @@ products:
price_max: 19900 price_max: 19900
price_min: 19900 price_min: 19900
price_varies: false price_varies: false
available: true available: true
tags: tags:
- season2006 - season2006
- beginner - beginner
- intermediate - intermediate
- freestyle - freestyle
- orange - orange
options: options:
- Length - Length
@@ -448,9 +448,9 @@ products:
- products/technine4.jpg - products/technine4.jpg
description: description:
2005 Technine Comic Series Description The Comic series was developed to be the ultimate progressive freestyle board in the Technine line. Dependable edge control and a perfect flex pattern for jumping in the park or out of bounds. Landins and progression will come easy with this board and it will help your riding progress to the next level. Street rails, park jibs, backcountry booters and park jumps, this board will do it all. 2005 Technine Comic Series Description The Comic series was developed to be the ultimate progressive freestyle board in the Technine line. Dependable edge control and a perfect flex pattern for jumping in the park or out of bounds. Landins and progression will come easy with this board and it will help your riding progress to the next level. Street rails, park jibs, backcountry booters and park jumps, this board will do it all.
variants: variants:
- *product-4-var-1 - *product-4-var-1
- &product-5 - &product-5
id: 5 id: 5
title: Burton Boots title: Burton Boots
@@ -461,11 +461,11 @@ products:
price_max: 11900 price_max: 11900
price_min: 11900 price_min: 11900
price_varies: false price_varies: false
available: true available: true
tags: tags:
- season2006 - season2006
- beginner - beginner
- intermediate - intermediate
- boots - boots
options: options:
- Color - Color
@@ -480,13 +480,13 @@ products:
- products/burton.jpg - products/burton.jpg
description: description:
The Burton boots are particularly well on snowboards. The very best thing about them is that the according picture is cubic. This makes testing in a Vision testing environment very easy. The Burton boots are particularly well on snowboards. The very best thing about them is that the according picture is cubic. This makes testing in a Vision testing environment very easy.
variants: variants:
- *product-5-var-1 - *product-5-var-1
- *product-5-var-2 - *product-5-var-2
- *product-5-var-3 - *product-5-var-3
- *product-5-var-4 - *product-5-var-4
- *product-5-var-5 - *product-5-var-5
- *product-5-var-6 - *product-5-var-6
- &product-6 - &product-6
id: 6 id: 6
@@ -498,17 +498,17 @@ products:
price_max: 2179500 price_max: 2179500
price_min: 2179500 price_min: 2179500
price_varies: false price_varies: false
available: true available: true
tags: tags:
- ducati - ducati
- superbike - superbike
- bike - bike
- street - street
- racing - racing
- performance - performance
options: options:
- Color - Color
compare_at_price: compare_at_price:
compare_at_price_max: 0 compare_at_price_max: 0
compare_at_price_min: 0 compare_at_price_min: 0
compare_at_price_varies: false compare_at_price_varies: false
@@ -520,7 +520,7 @@ products:
<h3>S PERFORMANCE</h3> <h3>S PERFORMANCE</h3>
<p>Producing 170hp (125kW) and with a dry weight of just 169kg (372.6lb), the new 1198 S now incorporates more World Superbike technology than ever before by taking the 1198 motor and adding top-of-the-range suspension, lightweight chassis components and a true racing-style traction control system designed for road use.</p> <p>Producing 170hp (125kW) and with a dry weight of just 169kg (372.6lb), the new 1198 S now incorporates more World Superbike technology than ever before by taking the 1198 motor and adding top-of-the-range suspension, lightweight chassis components and a true racing-style traction control system designed for road use.</p>
<p>The high performance, fully adjustable 43mm Öhlins forks, which sport low friction titanium nitride-treated fork sliders, respond effortlessly to every imperfection in the tarmac. Beyond their advanced engineering solutions, one of the most important characteristics of Öhlins forks is their ability to communicate the condition and quality of the tyre-to-road contact patch, a feature that puts every rider in superior control. The suspension set-up at the rear is complemented with a fully adjustable Öhlins rear shock equipped with a ride enhancing top-out spring and mounted to a single-sided swingarm for outstanding drive and traction. The front-to-rear Öhlins package is completed with a control-enhancing adjustable steering damper.</p> <p>The high performance, fully adjustable 43mm Öhlins forks, which sport low friction titanium nitride-treated fork sliders, respond effortlessly to every imperfection in the tarmac. Beyond their advanced engineering solutions, one of the most important characteristics of Öhlins forks is their ability to communicate the condition and quality of the tyre-to-road contact patch, a feature that puts every rider in superior control. The suspension set-up at the rear is complemented with a fully adjustable Öhlins rear shock equipped with a ride enhancing top-out spring and mounted to a single-sided swingarm for outstanding drive and traction. The front-to-rear Öhlins package is completed with a control-enhancing adjustable steering damper.</p>
variants: variants:
- *product-6-var-1 - *product-6-var-1
- &product-7 - &product-7
@@ -533,17 +533,17 @@ products:
price_max: 1900 price_max: 1900
price_min: 1900 price_min: 1900
price_varies: false price_varies: false
available: true available: true
tags: tags:
- shopify - shopify
- shirt - shirt
- apparel - apparel
- tshirt - tshirt
- clothing - clothing
options: options:
- Color - Color
- Size - Size
compare_at_price: compare_at_price:
compare_at_price_max: 0 compare_at_price_max: 0
compare_at_price_min: 0 compare_at_price_min: 0
compare_at_price_varies: false compare_at_price_varies: false
@@ -554,11 +554,11 @@ products:
description: description:
<p>High Quality Shopify Shirt. Wear your e-commerce solution with pride and attract attention anywhere you go.</p> <p>High Quality Shopify Shirt. Wear your e-commerce solution with pride and attract attention anywhere you go.</p>
<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p> <p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>
variants: variants:
- *product-7-var-1 - *product-7-var-1
- *product-7-var-2 - *product-7-var-2
- *product-7-var-3 - *product-7-var-3
- *product-7-var-4 - *product-7-var-4
- &product-8 - &product-8
id: 8 id: 8
@@ -570,11 +570,11 @@ products:
price_max: 5900 price_max: 5900
price_min: 5900 price_min: 5900
price_varies: false price_varies: false
available: true available: true
tags: tags:
- sweater - sweater
- hooded - hooded
- apparel - apparel
- clothing - clothing
options: options:
- Color - Color
@@ -591,11 +591,11 @@ products:
description: description:
<p>Extra comfortable zip up sweater. Durable quality, ideal for any outdoor activities.</p> <p>Extra comfortable zip up sweater. Durable quality, ideal for any outdoor activities.</p>
<p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p> <p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>
variants: variants:
- *product-8-var-1 - *product-8-var-1
- *product-8-var-2 - *product-8-var-2
- *product-8-var-3 - *product-8-var-3
- *product-8-var-4 - *product-8-var-4
- *product-8-var-5 - *product-8-var-5
- *product-8-var-6 - *product-8-var-6
@@ -609,11 +609,11 @@ products:
price_max: 552500 price_max: 552500
price_min: 499995 price_min: 499995
price_varies: true price_varies: true
available: true available: true
tags: tags:
- camera - camera
- slr - slr
- nikon - nikon
- professional - professional
options: options:
- Bundle - Bundle
@@ -626,21 +626,21 @@ products:
images: images:
- products/d3.jpg - products/d3.jpg
- products/d3_2.jpg - products/d3_2.jpg
- products/d3_3.jpg - products/d3_3.jpg
description: description:
<p>Flagship pro D-SLR with a 12.1-MP FX-format CMOS sensor, blazing 9 fps shooting at full FX resolution and low-noise performance up to 6400 ISO.</p> <p>Flagship pro D-SLR with a 12.1-MP FX-format CMOS sensor, blazing 9 fps shooting at full FX resolution and low-noise performance up to 6400 ISO.</p>
<p><strong>Nikon's original 12.1-megapixel FX-format (23.9 x 36mm) CMOS sensor:</strong> Couple Nikon's exclusive digital image processing system with the 12.1-megapixel FX-format and you'll get breathtakingly rich images while also reducing noise to unprecedented levels with even higher ISOs.</p> <p><strong>Nikon's original 12.1-megapixel FX-format (23.9 x 36mm) CMOS sensor:</strong> Couple Nikon's exclusive digital image processing system with the 12.1-megapixel FX-format and you'll get breathtakingly rich images while also reducing noise to unprecedented levels with even higher ISOs.</p>
<p><strong>Continuous shooting at up to 9 frames per second:</strong> At full FX resolution and up to 11fps in the DX crop mode, the D3 offers uncompromised shooting speeds for fast-action and sports photography.</p> <p><strong>Continuous shooting at up to 9 frames per second:</strong> At full FX resolution and up to 11fps in the DX crop mode, the D3 offers uncompromised shooting speeds for fast-action and sports photography.</p>
variants: variants:
- *product-9-var-1 - *product-9-var-1
- *product-9-var-2 - *product-9-var-2
- *product-9-var-3 - *product-9-var-3
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
# Line Items # Line Items
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
line_items: line_items:
- &line_item-1 - &line_item-1
id: 1 id: 1
@@ -661,7 +661,7 @@ line_items:
quantity: 2 quantity: 2
variant: *product-4-var-1 variant: *product-4-var-1
product: *product-4 product: *product-4
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
# Link Lists # Link Lists
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
@@ -715,7 +715,7 @@ links:
title: Catalog title: Catalog
url: /collections/all url: /collections/all
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
# Link Lists # Link Lists
@@ -727,7 +727,7 @@ link_lists:
title: 'Main Menu' title: 'Main Menu'
handle: 'main-menu' handle: 'main-menu'
links: links:
- *link-12 - *link-12
- *link-5 - *link-5
- *link-7 - *link-7
- *link-8 - *link-8
@@ -738,7 +738,7 @@ link_lists:
links: links:
- *link-5 - *link-5
- *link-6 - *link-6
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
# Collections # Collections
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
@@ -759,7 +759,7 @@ collections:
title: Arbor title: Arbor
handle: arbor handle: arbor
url: /collections/arbor url: /collections/arbor
products: products:
- *product-1 - *product-1
- *product-2 - *product-2
@@ -770,26 +770,26 @@ collections:
url: /collections/snowboards url: /collections/snowboards
description: description:
<p>This is a description for my <strong>Snowboards</strong> collection.</p> <p>This is a description for my <strong>Snowboards</strong> collection.</p>
products: products:
- *product-1 - *product-1
- *product-2 - *product-2
- *product-3 - *product-3
- *product-4 - *product-4
- &collection-4 - &collection-4
id: 4 id: 4
title: Items On Sale title: Items On Sale
handle: sale handle: sale
url: /collections/sale url: /collections/sale
products: products:
- *product-1 - *product-1
- &collection-5 - &collection-5
id: 5 id: 5
title: Paginated Sale title: Paginated Sale
handle: 'paginated-sale' handle: 'paginated-sale'
url: '/collections/paginated-sale' url: '/collections/paginated-sale'
products: products:
- *product-1 - *product-1
- *product-2 - *product-2
- *product-3 - *product-3
@@ -801,7 +801,7 @@ collections:
title: All products title: All products
handle: 'all' handle: 'all'
url: '/collections/all' url: '/collections/all'
products: products:
- *product-7 - *product-7
- *product-8 - *product-8
- *product-9 - *product-9
@@ -812,7 +812,7 @@ collections:
- *product-4 - *product-4
- *product-5 - *product-5
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
# Pages and Blogs # Pages and Blogs
# =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=- # =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
@@ -828,12 +828,12 @@ pages:
<p>Our retail store is located at <em>Rue d'Avignon 32, Avignon (Provence)</em>.</p> <p>Our retail store is located at <em>Rue d'Avignon 32, Avignon (Provence)</em>.</p>
<p><strong>Opening Hours:</strong><br />Monday through Friday: 9am - 6pm<br />Saturday: 10am - 3pm<br />Sunday: closed</p>" <p><strong>Opening Hours:</strong><br />Monday through Friday: 9am - 6pm<br />Saturday: 10am - 3pm<br />Sunday: closed</p>"
created_at: 2005-04-04 12:00 created_at: 2005-04-04 12:00
- &page-3 - &page-3
id: 2 id: 2
title: About Us title: About Us
handle: about-us handle: about-us
url: /pages/about-us url: /pages/about-us
author: Tobi author: Tobi
content: content:
"<p>Our company was founded in 1894 and we are since operating out of Avignon from the beautiful Provence.</p> "<p>Our company was founded in 1894 and we are since operating out of Avignon from the beautiful Provence.</p>
@@ -848,12 +848,12 @@ pages:
author: Tobi author: Tobi
content: "<ul><li>Your order is safe with us. Our checkout uses industry standard security to protect your information.</li><li>Your order will be billed immediately upon checkout.</li><li><b>ALL SALES ARE FINAL:</b> Defective or damaged product will be exchanged</li><li>All orders are processed expediently: usually in under 24 hours.</li></ul>" content: "<ul><li>Your order is safe with us. Our checkout uses industry standard security to protect your information.</li><li>Your order will be billed immediately upon checkout.</li><li><b>ALL SALES ARE FINAL:</b> Defective or damaged product will be exchanged</li><li>All orders are processed expediently: usually in under 24 hours.</li></ul>"
created_at: 2005-04-04 12:00 created_at: 2005-04-04 12:00
- &page-5 - &page-5
id: 4 id: 4
title: Shipping and Handling title: Shipping and Handling
handle: shipping handle: shipping
url: /pages/shipping url: /pages/shipping
author: Tobi author: Tobi
content: <p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p> content: <p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>
created_at: 2005-04-04 12:00 created_at: 2005-04-04 12:00
@@ -862,7 +862,7 @@ pages:
id: 5 id: 5
title: Frontpage title: Frontpage
handle: frontpage handle: frontpage
url: /pages/frontpage url: /pages/frontpage
author: Tobi author: Tobi
content: <p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p> content: <p>Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>
created_at: 2005-04-04 12:00 created_at: 2005-04-04 12:00
@@ -872,11 +872,11 @@ blogs:
handle: news handle: news
title: News title: News
url: /blogs/news url: /blogs/news
articles: articles:
- id: 3 - id: 3
title: 'Welcome to the new Foo Shop' title: 'Welcome to the new Foo Shop'
author: Daniel author: Daniel
content: <p><strong>Welcome to your Shopify store! The jaded Pixel crew is really glad you decided to take Shopify for a spin.</strong></p><p>To help you get you started with Shopify, here are a couple of tips regarding what you see on this page.</p><p>The text you see here is an article. To edit this article, create new articles or create new pages you can go to the <a href="/admin/pages">Blogs &amp; Pages</a> tab of the administration menu.</p><p>The Shopify t-shirt above is a product and selling products is what Shopify is all about. To edit this product, or create new products you can go to the <a href="/admin/products">Products Tab</a> in of the administration menu.</p><p>While you're looking around be sure to check out the <a href="/admin/collections">Collections</a> and <a href="/admin/links">Navigations</a> tabs and soon you will be well on your way to populating your site.</p><p>And of course don't forget to browse the <a href="admin/design/appearance/themes">theme gallery</a> to pick a new look for your shop!</p><p><strong>Shopify is in beta</strong><br />If you would like to make comments or suggestions please visit us in the <a href="http://forums.shopify.com/community">Shopify Forums</a> or drop us an <a href="mailto:feedback@shopify.com">email</a>.</p> content: <p><strong>Welcome to your Shopify store! The jaded Pixel crew is really glad you decided to take Shopify for a spin.</strong></p><p>To help you get you started with Shopify, here are a couple of tips regarding what you see on this page.</p><p>The text you see here is an article. To edit this article, create new articles or create new pages you can go to the <a href="/admin/pages">Blogs &amp; Pages</a> tab of the administration menu.</p><p>The Shopify t-shirt above is a product and selling products is what Shopify is all about. To edit this product, or create new products you can go to the <a href="/admin/products">Products Tab</a> in of the administration menu.</p><p>While you're looking around be sure to check out the <a href="/admin/collections">Collections</a> and <a href="/admin/links">Navigations</a> tabs and soon you will be well on your way to populating your site.</p><p>And of course don't forget to browse the <a href="admin/design/appearance/themes">theme gallery</a> to pick a new look for your shop!</p><p><strong>Shopify is in beta</strong><br />If you would like to make comments or suggestions please visit us in the <a href="http://forums.shopify.com/community">Shopify Forums</a> or drop us an <a href="mailto:feedback@shopify.com">email</a>.</p>
created_at: 2005-04-04 16:00 created_at: 2005-04-04 16:00
- id: 4 - id: 4
title: 'Breaking News: Restock on all sales products' title: 'Breaking News: Restock on all sales products'
@@ -884,7 +884,7 @@ blogs:
content: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. content: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.
created_at: 2005-04-04 12:00 created_at: 2005-04-04 12:00
articles_count: 2 articles_count: 2
- id: 2 - id: 2
handle: bigcheese-blog handle: bigcheese-blog
title: Bigcheese blog title: Bigcheese blog
@@ -896,7 +896,7 @@ blogs:
content: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. content: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.
created_at: 2005-04-04 16:00 created_at: 2005-04-04 16:00
comments: comments:
- -
id: 1 id: 1
author: John Smith author: John Smith
email: john@smith.com email: john@smith.com
@@ -905,7 +905,7 @@ blogs:
created_at: 2009-01-01 12:00 created_at: 2009-01-01 12:00
updated_at: 2009-02-01 12:00 updated_at: 2009-02-01 12:00
url: "" url: ""
- -
id: 2 id: 2
author: John Jones author: John Jones
email: john@jones.com email: john@jones.com
@@ -921,22 +921,22 @@ blogs:
created_at: 2005-04-06 12:00 created_at: 2005-04-06 12:00
comments: comments:
articles_count: 2 articles_count: 2
comments_enabled?: true comments_enabled?: true
comment_post_url: "" comment_post_url: ""
comments_count: 2 comments_count: 2
moderated?: true moderated?: true
- id: 3 - id: 3
handle: paginated-blog handle: paginated-blog
title: Paginated blog title: Paginated blog
url: /blogs/paginated-blog url: /blogs/paginated-blog
articles: articles:
- id: 6 - id: 6
title: 'One thing you probably did not know yet...' title: 'One thing you probably did not know yet...'
author: Justin author: Justin
content: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum. content: Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.
created_at: 2005-04-04 16:00 created_at: 2005-04-04 16:00
- id: 7 - id: 7
title: Fascinating title: Fascinating
author: Tobi author: Tobi

View File

@@ -3,9 +3,9 @@ module WeightFilter
def weight(grams) def weight(grams)
sprintf("%.2f", grams / 1000) sprintf("%.2f", grams / 1000)
end end
def weight_with_unit(grams) def weight_with_unit(grams)
"#{weight(grams)} kg" "#{weight(grams)} kg"
end end
end end

View File

@@ -4,7 +4,7 @@
<div class="article-body textile"> <div class="article-body textile">
{{ article.content }} {{ article.content }}
</div> </div>
</div> </div>
@@ -12,27 +12,27 @@
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
<div id="comments"> <div id="comments">
<h3>Comments</h3> <h3>Comments</h3>
<!-- List all comments --> <!-- List all comments -->
<ul id="comment-list"> <ul id="comment-list">
{% for comment in article.comments %} {% for comment in article.comments %}
<li> <li>
<div class="comment-details"> <div class="comment-details">
<span class="comment-author">{{ comment.author }}</span> said on <span class="comment-date">{{ comment.created_at | date: "%B %d, %Y" }}</span>: <span class="comment-author">{{ comment.author }}</span> said on <span class="comment-date">{{ comment.created_at | date: "%B %d, %Y" }}</span>:
</div> </div>
<div class="comment"> <div class="comment">
{{ comment.content }} {{ comment.content }}
</div> </div>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<!-- Comment Form --> <!-- Comment Form -->
<div id="comment-form"> <div id="comment-form">
{% form article %} {% form article %}
<h3>Leave a comment</h3> <h3>Leave a comment</h3>
<!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message --> <!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message -->
{% if form.posted_successfully? %} {% if form.posted_successfully? %}
{% if blog.moderated? %} {% if blog.moderated? %}
@@ -44,11 +44,11 @@
<div class="notice">Successfully posted your comment.</div> <div class="notice">Successfully posted your comment.</div>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% if form.errors %} {% if form.errors %}
<div class="notice error">Not all the fields have been filled out correctly!</div> <div class="notice error">Not all the fields have been filled out correctly!</div>
{% endif %} {% endif %}
<dl> <dl>
<dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt> <dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt>
<dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd> <dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd>
@@ -59,16 +59,16 @@
<dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt> <dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt>
<dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd> <dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd>
</dl> </dl>
{% if blog.moderated? %} {% if blog.moderated? %}
<p class="hint">comments have to be approved before showing up</p> <p class="hint">comments have to be approved before showing up</p>
{% endif %} {% endif %}
<input type="submit" value="Post comment" id="comment-submit" /> <input type="submit" value="Post comment" id="comment-submit" />
{% endform %} {% endform %}
</div> </div>
<!-- END Comment Form --> <!-- END Comment Form -->
</div> </div>
{% endif %} {% endif %}
<!-- END Comments --> <!-- END Comments -->

View File

@@ -1,33 +1,33 @@
<div id="page"> <div id="page">
<h2>{{page.title}}</h2> <h2>{{page.title}}</h2>
{% paginate blog.articles by 20 %} {% paginate blog.articles by 20 %}
{% for article in blog.articles %} {% for article in blog.articles %}
<div class="article"> <div class="article">
<div class="headline"> <div class="headline">
<h3 class="title"> <h3 class="title">
<a href="{{article.url}}">{{ article.title }}</a> <a href="{{article.url}}">{{ article.title }}</a>
</h3> </h3>
<h4 class="date">Posted on {{ article.created_at | date: "%B %d, '%y" }} by {{ article.author }}.</h4> <h4 class="date">Posted on {{ article.created_at | date: "%B %d, '%y" }} by {{ article.author }}.</h4>
</div> </div>
<div class="article-body textile"> <div class="article-body textile">
{{ article.content | strip_html | truncate: 250 }} {{ article.content | strip_html | truncate: 250 }}
</div> </div>
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
<p style="text-align: right"><a href="{{article.url}}#comments">{{ article.comments_count }} comments</a></p> <p style="text-align: right"><a href="{{article.url}}#comments">{{ article.comments_count }} comments</a></p>
{% endif %} {% endif %}
</div> </div>
{% endfor %} {% endfor %}
<div id="pagination"> <div id="pagination">
{{ paginate | default_pagination }} {{ paginate | default_pagination }}
</div> </div>
{% endpaginate %} {% endpaginate %}
</div> </div>

View File

@@ -1,66 +1,66 @@
<script type="text/javascript"> <script type="text/javascript">
function remove_item(id) { function remove_item(id) {
document.getElementById('updates_'+id).value = 0; document.getElementById('updates_'+id).value = 0;
document.getElementById('cartform').submit(); document.getElementById('cartform').submit();
} }
</script> </script>
<div> <div>
{% if cart.item_count == 0 %} {% if cart.item_count == 0 %}
<h4>Your shopping cart is looking rather empty...</h4> <h4>Your shopping cart is looking rather empty...</h4>
{% else %} {% else %}
<form action="/cart" method="post" id="cartform"> <form action="/cart" method="post" id="cartform">
<div id="cart"> <div id="cart">
<h3>You have {{ cart.item_count }} {{ cart.item_count | pluralize: 'product', 'products' }} in here!</h3> <h3>You have {{ cart.item_count }} {{ cart.item_count | pluralize: 'product', 'products' }} in here!</h3>
<ul id="line-items"> <ul id="line-items">
{% for item in cart.items %} {% for item in cart.items %}
<li id="item-{{item.id}}" class="clearfix"> <li id="item-{{item.id}}" class="clearfix">
<div class="thumb"> <div class="thumb">
<div class="prodimage"> <div class="prodimage">
<a href="{{item.product.url}}" title="View {{item.title}} Page"><img src="{{item.product.featured_image | product_img_url: 'thumb' }}" alt="{{item.title | escape }}" /></a> <a href="{{item.product.url}}" title="View {{item.title}} Page"><img src="{{item.product.featured_image | product_img_url: 'thumb' }}" alt="{{item.title | escape }}" /></a>
</div></div> </div></div>
<h3 style="padding-right: 150px"> <h3 style="padding-right: 150px">
<a href="{{item.product.url}}" title="View {{item.title | escape }} Page"> <a href="{{item.product.url}}" title="View {{item.title | escape }} Page">
{{ item.title }} {{ item.title }}
{% if item.variant.available == true %} {% if item.variant.available == true %}
({{item.variant.title}}) ({{item.variant.title}})
{% endif %} {% endif %}
</a> </a>
</h3> </h3>
<small class="itemcost">Costs {{ item.price | money }} each, <span class="money">{{item.line_price | money }}</span> total.</small> <small class="itemcost">Costs {{ item.price | money }} each, <span class="money">{{item.line_price | money }}</span> total.</small>
<p class="right"> <p class="right">
<label for="updates">How many? </label> <label for="updates">How many? </label>
<input type="text" size="4" name="updates[{{item.variant.id}}]" id="updates_{{item.variant.id}}" value="{{item.quantity}}" onfocus="this.select();"/><br /> <input type="text" size="4" name="updates[{{item.variant.id}}]" id="updates_{{item.variant.id}}" value="{{item.quantity}}" onfocus="this.select();"/><br />
<a href="#" onclick="remove_item({{item.variant.id}}); return false;" class="remove"><img style="padding:15px 0 0 0;margin:0;" src="{{ 'delete.gif' | asset_url }}" /></a> <a href="#" onclick="remove_item({{item.variant.id}}); return false;" class="remove"><img style="padding:15px 0 0 0;margin:0;" src="{{ 'delete.gif' | asset_url }}" /></a>
</p> </p>
</li> </li>
{% endfor %} {% endfor %}
<li id="total"> <li id="total">
<input type="image" id="update-cart" name="update" value="Update My Cart" src="{{ 'update.gif' | asset_url }}" /> <input type="image" id="update-cart" name="update" value="Update My Cart" src="{{ 'update.gif' | asset_url }}" />
Subtotal: Subtotal:
<span class="money">{{ cart.total_price | money_with_currency }}</span> <span class="money">{{ cart.total_price | money_with_currency }}</span>
</li> </li>
</ul> </ul>
</div> </div>
<div class="info"> <div class="info">
<input type="image" value="Checkout!" name="checkout" src="{{ 'checkout.gif' | asset_url }}" /> <input type="image" value="Checkout!" name="checkout" src="{{ 'checkout.gif' | asset_url }}" />
</div> </div>
{% if additional_checkout_buttons %} {% if additional_checkout_buttons %}
<div class="additional-checkout-buttons"> <div class="additional-checkout-buttons">
<p>- or -</p> <p>- or -</p>
{{ content_for_additional_checkout_buttons }} {{ content_for_additional_checkout_buttons }}
</div> </div>
{% endif %} {% endif %}
</form> </form>
{% endif %} {% endif %}
</div> </div>

View File

@@ -1,22 +1,22 @@
{% paginate collection.products by 20 %} {% paginate collection.products by 20 %}
<ul id="product-collection"> <ul id="product-collection">
{% for product in collection.products %} {% for product in collection.products %}
<li class="singleproduct clearfix"> <li class="singleproduct clearfix">
<div class="small"> <div class="small">
<div class="prodimage"><a href="{{product.url}}"><img src="{{ product.featured_image | product_img_url: 'small' }}" /></a></div> <div class="prodimage"><a href="{{product.url}}"><img src="{{ product.featured_image | product_img_url: 'small' }}" /></a></div>
</div> </div>
<div class="description"> <div class="description">
<h3><a href="{{product.url}}">{{product.title}}</a></h3> <h3><a href="{{product.url}}">{{product.title}}</a></h3>
<p>{{ product.description | strip_html | truncatewords: 35 }}</p> <p>{{ product.description | strip_html | truncatewords: 35 }}</p>
<p class="money">{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</p> <p class="money">{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</p>
</div> </div>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<div id="pagination"> <div id="pagination">
{{ paginate | default_pagination }} {{ paginate | default_pagination }}
</div> </div>
{% endpaginate %} {% endpaginate %}

View File

@@ -1,47 +1,47 @@
<div id="frontproducts"><div id="frontproducts-top"><div id="frontproducts-bottom"> <div id="frontproducts"><div id="frontproducts-top"><div id="frontproducts-bottom">
<h2 style="display: none;">Featured Items</h2> <h2 style="display: none;">Featured Items</h2>
{% for product in collections.frontpage.products limit:1 offset:0 %} {% for product in collections.frontpage.products limit:1 offset:0 %}
<div class="productmain"> <div class="productmain">
<a href="{{ product.url }}"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ product.url }}"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a>
<h3><a href="{{ product.url }}">{{ product.title }}</a></h3> <h3><a href="{{ product.url }}">{{ product.title }}</a></h3>
<div class="description">{{ product.description | strip_html | truncatewords: 18 }}</div> <div class="description">{{ product.description | strip_html | truncatewords: 18 }}</div>
<p class="money">{{ product.price_min | money }}</p> <p class="money">{{ product.price_min | money }}</p>
</div> </div>
{% endfor %} {% endfor %}
{% for product in collections.frontpage.products offset:1 %} {% for product in collections.frontpage.products offset:1 %}
<div class="product"> <div class="product">
<a href="{{ product.url }}"><img src="{{ product.featured_image | product_img_url: 'thumb' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ product.url }}"><img src="{{ product.featured_image | product_img_url: 'thumb' }}" alt="{{ product.title | escape }}" /></a>
<h3><a href="{{ product.url }}">{{ product.title }}</a></h3> <h3><a href="{{ product.url }}">{{ product.title }}</a></h3>
<p class="money">{{ product.price_min | money }}</p> <p class="money">{{ product.price_min | money }}</p>
</div> </div>
{% endfor %} {% endfor %}
</div></div></div> </div></div></div>
<div id="mainarticle"> <div id="mainarticle">
{% assign article = pages.frontpage %} {% assign article = pages.frontpage %}
{% if article.content != "" %} {% if article.content != "" %}
<h2>{{ article.title }}</h2> <h2>{{ article.title }}</h2>
<div class="article-body textile"> <div class="article-body textile">
{{ article.content }} {{ article.content }}
</div> </div>
{% else %} {% else %}
<div class="article-body textile"> <div class="article-body textile">
In <em>Admin &gt; Blogs &amp; Pages</em>, create a page with the handle <strong><code>frontpage</code></strong> and it will show up here.<br /> In <em>Admin &gt; Blogs &amp; Pages</em>, create a page with the handle <strong><code>frontpage</code></strong> and it will show up here.<br />
{{ "Learn more about handles" | link_to "http://wiki.shopify.com/Handle" }} {{ "Learn more about handles" | link_to "http://wiki.shopify.com/Handle" }}
</div> </div>
{% endif %} {% endif %}
</div> </div>
<br style="clear: both;" /> <br style="clear: both;" />
<div id="articles"> <div id="articles">
{% for article in blogs.news.articles offset:1 %} {% for article in blogs.news.articles offset:1 %}
<div class="article"> <div class="article">
<h2>{{ article.title }}</h2> <h2>{{ article.title }}</h2>
<div class="article-body textile"> <div class="article-body textile">
{{ article.content }} {{ article.content }}
</div> </div>
</div> </div>
{% endfor %} {% endfor %}
</div> </div>

View File

@@ -5,4 +5,4 @@
{{page.content}} {{page.content}}
</div> </div>
</div> </div>

View File

@@ -1,68 +1,68 @@
<div id="productpage"> <div id="productpage">
<div id="productimages"><div id="productimages-top"><div id="productimages-bottom"> <div id="productimages"><div id="productimages-top"><div id="productimages-bottom">
{% for image in product.images %} {% for image in product.images %}
{% if forloop.first %} {% if forloop.first %}
<a href="{{ image | product_img_url: 'large' }}" class="productimage" rel="lightbox"> <a href="{{ image | product_img_url: 'large' }}" class="productimage" rel="lightbox">
<img src="{{ image | product_img_url: 'medium'}}" alt="{{product.title | escape }}" /> <img src="{{ image | product_img_url: 'medium'}}" alt="{{product.title | escape }}" />
</a> </a>
{% else %} {% else %}
<a href="{{ image | product_img_url: 'large' }}" class="productimage-small" rel="lightbox"> <a href="{{ image | product_img_url: 'large' }}" class="productimage-small" rel="lightbox">
<img src="{{ image | product_img_url: 'small'}}" alt="{{product.title | escape }}" /> <img src="{{ image | product_img_url: 'small'}}" alt="{{product.title | escape }}" />
</a> </a>
{% endif %} {% endif %}
{% endfor %} {% endfor %}
</div></div></div> </div></div></div>
<h2>{{ product.title }}</h2> <h2>{{ product.title }}</h2>
<ul id="details" class="hlist"> <ul id="details" class="hlist">
<li>Vendor: {{ product.vendor | link_to_vendor }}</li> <li>Vendor: {{ product.vendor | link_to_vendor }}</li>
<li>Type: {{ product.type | link_to_type }}</li> <li>Type: {{ product.type | link_to_type }}</li>
</ul> </ul>
<small>{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</small> <small>{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</small>
<div id="variant-add"> <div id="variant-add">
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<select id="variant-select" name="id" class="product-info-options"> <select id="variant-select" name="id" class="product-info-options">
{% for variant in product.variants %} {% for variant in product.variants %}
<option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option> <option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option>
{% endfor %} {% endfor %}
</select> </select>
<div id="price-field" class="price"></div> <div id="price-field" class="price"></div>
<div style="text-align:center;"><input type="image" name="add" value="Add to Cart" id="add" src="{{ 'addtocart.gif' | asset_url }}" /></div> <div style="text-align:center;"><input type="image" name="add" value="Add to Cart" id="add" src="{{ 'addtocart.gif' | asset_url }}" /></div>
</form> </form>
</div> </div>
<div class="description textile"> <div class="description textile">
{{ product.description }} {{ product.description }}
</div> </div>
</div> </div>
<script type="text/javascript"> <script type="text/javascript">
<!-- <!--
// prototype callback for multi variants dropdown selector // prototype callback for multi variants dropdown selector
var selectCallback = function(variant, selector) { var selectCallback = function(variant, selector) {
if (variant && variant.available == true) { if (variant && variant.available == true) {
// selected a valid variant // selected a valid variant
$('add').removeClassName('disabled'); // remove unavailable class from add-to-cart button $('add').removeClassName('disabled'); // remove unavailable class from add-to-cart button
$('add').disabled = false; // reenable add-to-cart button $('add').disabled = false; // reenable add-to-cart button
$('price-field').innerHTML = Shopify.formatMoney(variant.price, "{{shop.money_with_currency_format}}"); // update price field $('price-field').innerHTML = Shopify.formatMoney(variant.price, "{{shop.money_with_currency_format}}"); // update price field
} else { } else {
// variant doesn't exist // variant doesn't exist
$('add').addClassName('disabled'); // set add-to-cart button to unavailable class $('add').addClassName('disabled'); // set add-to-cart button to unavailable class
$('add').disabled = true; // disable add-to-cart button $('add').disabled = true; // disable add-to-cart button
$('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message $('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message
} }
}; };
// initialize multi selector for product // initialize multi selector for product
Event.observe(document, 'dom:loaded', function() { Event.observe(document, 'dom:loaded', function() {
new Shopify.OptionSelectors("variant-select", { product: {{ product | json }}, onVariantSelected: selectCallback }); new Shopify.OptionSelectors("variant-select", { product: {{ product | json }}, onVariantSelected: selectCallback });
}); });
--> -->
</script> </script>

View File

@@ -1,105 +1,105 @@
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en"> <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
<head> <head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
<title>{{shop.name}} - {{page_title}}</title> <title>{{shop.name}} - {{page_title}}</title>
{{ 'textile.css' | global_asset_url | stylesheet_tag }} {{ 'textile.css' | global_asset_url | stylesheet_tag }}
{{ 'lightbox/v204/lightbox.css' | global_asset_url | stylesheet_tag }} {{ 'lightbox/v204/lightbox.css' | global_asset_url | stylesheet_tag }}
{{ 'prototype/1.6/prototype.js' | global_asset_url | script_tag }} {{ 'prototype/1.6/prototype.js' | global_asset_url | script_tag }}
{{ 'scriptaculous/1.8.2/scriptaculous.js' | global_asset_url | script_tag }} {{ 'scriptaculous/1.8.2/scriptaculous.js' | global_asset_url | script_tag }}
{{ 'lightbox/v204/lightbox.js' | global_asset_url | script_tag }} {{ 'lightbox/v204/lightbox.js' | global_asset_url | script_tag }}
{{ 'option_selection.js' | shopify_asset_url | script_tag }} {{ 'option_selection.js' | shopify_asset_url | script_tag }}
{{ 'layout.css' | asset_url | stylesheet_tag }} {{ 'layout.css' | asset_url | stylesheet_tag }}
{{ 'shop.js' | asset_url | script_tag }} {{ 'shop.js' | asset_url | script_tag }}
{{ content_for_header }} {{ content_for_header }}
</head> </head>
<body id="page-{{template}}"> <body id="page-{{template}}">
<p class="hide"><a href="#rightsiders">Skip to navigation.</a></p> <p class="hide"><a href="#rightsiders">Skip to navigation.</a></p>
<!-- mini cart --> <!-- mini cart -->
{% if cart.item_count > 0 %} {% if cart.item_count > 0 %}
<div id="minicart" style="display:none;"><div id="minicart-inner"> <div id="minicart" style="display:none;"><div id="minicart-inner">
<div id="minicart-items"> <div id="minicart-items">
<h2>There {{ cart.item_count | pluralize: 'is', 'are' }} {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }} in <a href="/cart" title="View your cart">your cart</a>!</h2><h4 style="font-size: 16px; margin: 0 0 10px 0; padding: 0;">Your subtotal is {{ cart.total_price | money }}.</h4> <h2>There {{ cart.item_count | pluralize: 'is', 'are' }} {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }} in <a href="/cart" title="View your cart">your cart</a>!</h2><h4 style="font-size: 16px; margin: 0 0 10px 0; padding: 0;">Your subtotal is {{ cart.total_price | money }}.</h4>
{% for item in cart.items %} {% for item in cart.items %}
<div class="thumb"> <div class="thumb">
<div class="prodimage"><a href="{{item.product.url}}" onMouseover="tooltip('{{ item.quantity }} x {{ item.title }} ({{ item.variant.title }})', 200)"; onMouseout="hidetooltip()"><img src="{{ item.product.featured_image | product_img_url: 'thumb' }}" /></a></div> <div class="prodimage"><a href="{{item.product.url}}" onMouseover="tooltip('{{ item.quantity }} x {{ item.title }} ({{ item.variant.title }})', 200)"; onMouseout="hidetooltip()"><img src="{{ item.product.featured_image | product_img_url: 'thumb' }}" /></a></div>
</div> </div>
{% endfor %} {% endfor %}
</div> </div>
<br style="clear:both;" /> <br style="clear:both;" />
</div></div> </div></div>
{% endif %} {% endif %}
<div id="container"> <div id="container">
<div id="header"> <div id="header">
<!-- Begin Header --> <!-- Begin Header -->
<h1 id="logo"><a href="/" title="Go Home">{{shop.name}}</a></h1> <h1 id="logo"><a href="/" title="Go Home">{{shop.name}}</a></h1>
<div id="cartlinks"> <div id="cartlinks">
{% if cart.item_count > 0 %} {% if cart.item_count > 0 %}
<h2 id="cartcount"><a href="/cart" onMouseover="tooltip('There {{ cart.item_count | pluralize: 'is', 'are' }} {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }} in your cart!', 200)"; onMouseout="hidetooltip()">{{ cart.item_count }} {{ cart.item_count | pluralize: 'thing', 'things' }}!</a></h2> <h2 id="cartcount"><a href="/cart" onMouseover="tooltip('There {{ cart.item_count | pluralize: 'is', 'are' }} {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }} in your cart!', 200)"; onMouseout="hidetooltip()">{{ cart.item_count }} {{ cart.item_count | pluralize: 'thing', 'things' }}!</a></h2>
<a href="/cart" id="minicartswitch" onclick="superSwitch(this, 'minicart', 'Close Mini Cart'); return false;" id="cartswitch">View Mini Cart ({{ cart.total_price | money }})</a> <a href="/cart" id="minicartswitch" onclick="superSwitch(this, 'minicart', 'Close Mini Cart'); return false;" id="cartswitch">View Mini Cart ({{ cart.total_price | money }})</a>
{% endif %} {% endif %}
</div> </div>
<!-- End Header --> <!-- End Header -->
</div> </div>
<hr /> <hr />
<div id="main"> <div id="main">
<div id="content"> <div id="content">
<div id="innercontent"> <div id="innercontent">
{{ content_for_layout }} {{ content_for_layout }}
</div> </div>
</div> </div>
<hr /> <hr />
<div id="rightsiders"> <div id="rightsiders">
<ul class="rightlinks"> <ul class="rightlinks">
{% for link in linklists.main-menu.links %} {% for link in linklists.main-menu.links %}
<li>{{ link.title | link_to: link.url }}</li> <li>{{ link.title | link_to: link.url }}</li>
{% endfor %} {% endfor %}
</ul> </ul>
{% if tags %} {% if tags %}
<ul class="rightlinks"> <ul class="rightlinks">
{% for tag in collection.tags %} {% for tag in collection.tags %}
<li><span class="add-link">{{ '+' | link_to_add_tag: tag }}</span>{{ tag | highlight_active_tag | link_to_tag: tag }}</li> <li><span class="add-link">{{ '+' | link_to_add_tag: tag }}</span>{{ tag | highlight_active_tag | link_to_tag: tag }}</li>
{% endfor %} {% endfor %}
</ul> </ul>
{% endif %} {% endif %}
<ul class="rightlinks"> <ul class="rightlinks">
{% for link in linklists.footer.links %} {% for link in linklists.footer.links %}
<li>{{ link.title | link_to: link.url }}</li> <li>{{ link.title | link_to: link.url }}</li>
{% endfor %} {% endfor %}
</ul> </ul>
</div> </div>
<hr /><br style="clear:both;" /> <hr /><br style="clear:both;" />
<div id="footer"> <div id="footer">
<div class="footerinner"> <div class="footerinner">
All prices are in {{ shop.currency }}. All prices are in {{ shop.currency }}.
Powered by <a href="http://www.shopify.com" title="Shopify, Hosted E-Commerce">Shopify</a>. Powered by <a href="http://www.shopify.com" title="Shopify, Hosted E-Commerce">Shopify</a>.
</div> </div>
</div> </div>
</div> </div>
</div> </div>
<div id="tooltip"></div> <div id="tooltip"></div>
<img id="pointer" src="{{ 'arrow2.gif' | asset_url }}" /> <img id="pointer" src="{{ 'arrow2.gif' | asset_url }}" />
</body> </body>
</html> </html>

View File

@@ -4,7 +4,7 @@
<div class="article-body textile"> <div class="article-body textile">
{{ article.content }} {{ article.content }}
</div> </div>
</div> </div>
@@ -12,7 +12,7 @@
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
<div id="comments"> <div id="comments">
<h3>Comments</h3> <h3>Comments</h3>
<!-- List all comments --> <!-- List all comments -->
<ul id="comment-list"> <ul id="comment-list">
{% for comment in article.comments %} {% for comment in article.comments %}
@@ -20,19 +20,19 @@
<div class="comment"> <div class="comment">
{{ comment.content }} {{ comment.content }}
</div> </div>
<div class="comment-details"> <div class="comment-details">
Posted by {{ comment.author }} on {{ comment.created_at | date: "%B %d, %Y" }} Posted by {{ comment.author }} on {{ comment.created_at | date: "%B %d, %Y" }}
</div> </div>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<!-- Comment Form --> <!-- Comment Form -->
<div id="comment-form"> <div id="comment-form">
{% form article %} {% form article %}
<h3>Leave a comment</h3> <h3>Leave a comment</h3>
<!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message --> <!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message -->
{% if form.posted_successfully? %} {% if form.posted_successfully? %}
{% if blog.moderated? %} {% if blog.moderated? %}
@@ -44,11 +44,11 @@
<div class="notice">Successfully posted your comment.</div> <div class="notice">Successfully posted your comment.</div>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% if form.errors %} {% if form.errors %}
<div class="notice error">Not all the fields have been filled out correctly!</div> <div class="notice error">Not all the fields have been filled out correctly!</div>
{% endif %} {% endif %}
<dl> <dl>
<dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt> <dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt>
<dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd> <dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd>
@@ -59,16 +59,16 @@
<dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt> <dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt>
<dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd> <dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd>
</dl> </dl>
{% if blog.moderated? %} {% if blog.moderated? %}
<p class="hint">comments have to be approved before showing up</p> <p class="hint">comments have to be approved before showing up</p>
{% endif %} {% endif %}
<input type="submit" value="Post comment" id="comment-submit" /> <input type="submit" value="Post comment" id="comment-submit" />
{% endform %} {% endform %}
</div> </div>
<!-- END Comment Form --> <!-- END Comment Form -->
</div> </div>
{% endif %} {% endif %}
<!-- END Comments --> <!-- END Comments -->

View File

@@ -1,13 +1,13 @@
<div id="blog-page"> <div id="blog-page">
<h2 class="heading-shaded">{{page.title}}</h2> <h2 class="heading-shaded">{{page.title}}</h2>
{% for article in blog.articles %} {% for article in blog.articles %}
<h4> <h4>
{{ article.created_at | date: '%d %b' }} {{ article.created_at | date: '%d %b' }}
<a href="{{article.url}}">{{ article.title }}</a> <a href="{{article.url}}">{{ article.title }}</a>
</h4> </h4>
{{ article.content }} {{ article.content }}
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
<p><a href="{{article.url}}#comments">{{ article.comments_count }} comments</a></p> <p><a href="{{article.url}}#comments">{{ article.comments_count }} comments</a></p>
{% endif %} {% endif %}
{% endfor %} {% endfor %}
</div> </div>

View File

@@ -7,9 +7,9 @@
<div id="cart-page"> <div id="cart-page">
{% if cart.item_count == 0 %} {% if cart.item_count == 0 %}
<p>Your shopping cart is empty...</p> <p>Your shopping cart is empty...</p>
<p><a href="/"><img src="{{ 'continue_shopping_icon.gif' | asset_url }}" alt="Continue shopping"/></a><p> <p><a href="/"><img src="{{ 'continue_shopping_icon.gif' | asset_url }}" alt="Continue shopping"/></a><p>
{% else %} {% else %}
<form action="/cart" method="post" id="cart"> <form action="/cart" method="post" id="cart">
@@ -26,21 +26,21 @@
{% for item in cart.items %} {% for item in cart.items %}
<tr class="{% cycle 'odd', 'even' %}"> <tr class="{% cycle 'odd', 'even' %}">
<td class="short">{{ item.product.featured_image | product_img_url: 'thumb' | img_tag }}</td> <td class="short">{{ item.product.featured_image | product_img_url: 'thumb' | img_tag }}</td>
<td><a href="{{item.product.url}}">{{ item.title }}</a></td> <td><a href="{{item.product.url}}">{{ item.title }}</a></td>
<td class="short"><input type="text" class="quantity" name="updates[{{item.variant.id}}]" id="updates_{{item.variant.id}}" value="{{item.quantity}}" onfocus="this.select();"/></td> <td class="short"><input type="text" class="quantity" name="updates[{{item.variant.id}}]" id="updates_{{item.variant.id}}" value="{{item.quantity}}" onfocus="this.select();"/></td>
<td class="cart-price">{{ item.price | money }}</td> <td class="cart-price">{{ item.price | money }}</td>
<td class="cart-price">{{item.line_price | money }}</td> <td class="cart-price">{{item.line_price | money }}</td>
<td class="short"><a href="#" onclick="remove_item({{item.variant.id}}); return false;" class="remove"><img src="{{ 'cancel_icon.gif' | asset_url }}" alt="Remove" /></a></td> <td class="short"><a href="#" onclick="remove_item({{item.variant.id}}); return false;" class="remove"><img src="{{ 'cancel_icon.gif' | asset_url }}" alt="Remove" /></a></td>
</tr> </tr>
{% endfor %} {% endfor %}
</table> </table>
<p class="updatebtn"><input type="image" value="Update Cart" name="update" src="{{ 'update_icon.gif' | asset_url }}" alt="Update" /></p> <p class="updatebtn"><input type="image" value="Update Cart" name="update" src="{{ 'update_icon.gif' | asset_url }}" alt="Update" /></p>
<p class="subtotal"> <p class="subtotal">
<strong>Subtotal:</strong> {{cart.total_price | money_with_currency }} <strong>Subtotal:</strong> {{cart.total_price | money_with_currency }}
</p> </p>
<p class="checkout"><input type="image" src="{{ 'checkout_icon.gif' | asset_url }}" alt="Proceed to Checkout" value="Proceed to Checkout" name="checkout" /></p> <p class="checkout"><input type="image" src="{{ 'checkout_icon.gif' | asset_url }}" alt="Proceed to Checkout" value="Proceed to Checkout" name="checkout" /></p>
{% if additional_checkout_buttons %} {% if additional_checkout_buttons %}
<div class="additional-checkout-buttons"> <div class="additional-checkout-buttons">
<p>- or -</p> <p>- or -</p>
{{ content_for_additional_checkout_buttons }} {{ content_for_additional_checkout_buttons }}
@@ -48,7 +48,7 @@
{% endif %} {% endif %}
</form> </form>
{% endif %} {% endif %}
</div> </div>

View File

@@ -1,29 +1,29 @@
<div id="collection-page"> <div id="collection-page">
{% if collection.description %} {% if collection.description %}
<div id="collection-description" class="textile">{{ collection.description }}</div> <div id="collection-description" class="textile">{{ collection.description }}</div>
{% endif %} {% endif %}
{% paginate collection.products by 20 %} {% paginate collection.products by 20 %}
<ul id="product-collection"> <ul id="product-collection">
{% for product in collection.products %} {% for product in collection.products %}
<li class="single-product clearfix"> <li class="single-product clearfix">
<div class="small"> <div class="small">
<div class="prod-image"><a href="{{product.url}}"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a></div> <div class="prod-image"><a href="{{product.url}}"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a></div>
</div> </div>
<div class="prod-list-description"> <div class="prod-list-description">
<h3><a href="{{product.url}}">{{product.title}}</a></h3> <h3><a href="{{product.url}}">{{product.title}}</a></h3>
<p>{{ product.description | strip_html | truncatewords: 35 }}</p> <p>{{ product.description | strip_html | truncatewords: 35 }}</p>
<p class="prd-price">{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</p> <p class="prd-price">{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</p>
</div> </div>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<div id="pagination"> <div id="pagination">
{{ paginate | default_pagination }} {{ paginate | default_pagination }}
</div> </div>
{% endpaginate %} {% endpaginate %}
</div> </div>

View File

@@ -1,32 +1,32 @@
<div id="home-page"> <div id="home-page">
<h3 class="heading-shaded">Featured products...</h3> <h3 class="heading-shaded">Featured products...</h3>
<div class="featured-prod-row clearfix"> <div class="featured-prod-row clearfix">
{% for product in collections.frontpage.products %} {% for product in collections.frontpage.products %}
<div class="featured-prod-item"> <div class="featured-prod-item">
<p> <p>
<a href="{{product.url}}"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a> <a href="{{product.url}}"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</p> </p>
<h4><a href="{{product.url}}">{{product.title}}</a></h4> <h4><a href="{{product.url}}">{{product.title}}</a></h4>
{% if product.compare_at_price %} {% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %} {% if product.price_min != product.compare_at_price %}
<p class="prd-price">Was:<del>{{product.compare_at_price | money}}</del></p> <p class="prd-price">Was:<del>{{product.compare_at_price | money}}</del></p>
<p class="prd-price"><ins>Now: {{product.price_min | money}}</ins></p> <p class="prd-price"><ins>Now: {{product.price_min | money}}</ins></p>
{% endif %} {% endif %}
{% else %} {% else %}
<p class="prd-price"><ins>{{product.price_min | money}}</ins></p> <p class="prd-price"><ins>{{product.price_min | money}}</ins></p>
{% endif %} {% endif %}
</div> </div>
{% endfor %} {% endfor %}
</div> </div>
<div id="articles"> <div id="articles">
{% assign article = pages.frontpage %} {% assign article = pages.frontpage %}
{% if article.content != "" %} {% if article.content != "" %}
<h3>{{ article.title }}</h3> <h3>{{ article.title }}</h3>
{{ article.content }} {{ article.content }}
{% else %} {% else %}
In <em>Admin &gt; Blogs &amp; Pages</em>, create a page with the handle <strong><code>frontpage</code></strong> and it will show up here.<br /> In <em>Admin &gt; Blogs &amp; Pages</em>, create a page with the handle <strong><code>frontpage</code></strong> and it will show up here.<br />
{{ "Learn more about handles" | link_to "http://wiki.shopify.com/Handle" }} {{ "Learn more about handles" | link_to "http://wiki.shopify.com/Handle" }}
{% endif %} {% endif %}
</div> </div>
</div> </div>

View File

@@ -1,4 +1,4 @@
<div id="single-page"> <div id="single-page">
<h2 class="heading-shaded">{{page.title}}</h2> <h2 class="heading-shaded">{{page.title}}</h2>
{{ page.content }} {{ page.content }}
</div> </div>

View File

@@ -1,75 +1,75 @@
<div id="product-page"> <div id="product-page">
<h2 class="heading-shaded">{{ product.title }}</h2> <h2 class="heading-shaded">{{ product.title }}</h2>
<div id="product-details"> <div id="product-details">
<div id="product-images"> <div id="product-images">
{% for image in product.images %} {% for image in product.images %}
{% if forloop.first %} {% if forloop.first %}
<a href="{{ image | product_img_url: 'large' }}" class="product-image" rel="lightbox[ product]" title=""> <a href="{{ image | product_img_url: 'large' }}" class="product-image" rel="lightbox[ product]" title="">
<img src="{{ image | product_img_url: 'medium'}}" alt="{{product.title | escape }}" /> <img src="{{ image | product_img_url: 'medium'}}" alt="{{product.title | escape }}" />
</a> </a>
{% else %} {% else %}
<a href="{{ image | product_img_url: 'large' }}" class="product-image-small" rel="lightbox[ product]" title=""> <a href="{{ image | product_img_url: 'large' }}" class="product-image-small" rel="lightbox[ product]" title="">
<img src="{{ image | product_img_url: 'small'}}" alt="{{product.title | escape }}" /> <img src="{{ image | product_img_url: 'small'}}" alt="{{product.title | escape }}" />
</a> </a>
{% endif %} {% endif %}
{% endfor %} {% endfor %}
</div> </div>
<ul id="product-info"> <ul id="product-info">
<li>Vendor: {{ product.vendor | link_to_vendor }}</li> <li>Vendor: {{ product.vendor | link_to_vendor }}</li>
<li>Type: {{ product.type | link_to_type }}</li> <li>Type: {{ product.type | link_to_type }}</li>
</ul> </ul>
<small>{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</small> <small>{{ product.price_min | money }}{% if product.price_varies %} - {{ product.price_max | money }}{% endif %}</small>
<div id="product-options"> <div id="product-options">
{% if product.available %} {% if product.available %}
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<select id="product-select" name='id'> <select id="product-select" name='id'>
{% for variant in product.variants %} {% for variant in product.variants %}
<option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option> <option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option>
{% endfor %} {% endfor %}
</select> </select>
<div id="price-field"></div> <div id="price-field"></div>
<div class="add-to-cart"><input type="image" name="add" value="Add to Cart" id="add" src="{{ 'add-to-cart.gif' | asset_url }}" /></div> <div class="add-to-cart"><input type="image" name="add" value="Add to Cart" id="add" src="{{ 'add-to-cart.gif' | asset_url }}" /></div>
</form> </form>
{% else %} {% else %}
<span>Sold Out!</span> <span>Sold Out!</span>
{% endif %} {% endif %}
</div> </div>
<div class="product-description"> <div class="product-description">
{{ product.description }} {{ product.description }}
</div> </div>
</div> </div>
</div> </div>
<script type="text/javascript"> <script type="text/javascript">
<!-- <!--
// mootools callback for multi variants dropdown selector // mootools callback for multi variants dropdown selector
var selectCallback = function(variant, selector) { var selectCallback = function(variant, selector) {
if (variant && variant.available == true) { if (variant && variant.available == true) {
// selected a valid variant // selected a valid variant
$('add').removeClass('disabled'); // remove unavailable class from add-to-cart button $('add').removeClass('disabled'); // remove unavailable class from add-to-cart button
$('add').disabled = false; // reenable add-to-cart button $('add').disabled = false; // reenable add-to-cart button
$('price-field').innerHTML = Shopify.formatMoney(variant.price, "{{shop.money_with_currency_format}}"); // update price field $('price-field').innerHTML = Shopify.formatMoney(variant.price, "{{shop.money_with_currency_format}}"); // update price field
} else { } else {
// variant doesn't exist // variant doesn't exist
$('add').addClass('disabled'); // set add-to-cart button to unavailable class $('add').addClass('disabled'); // set add-to-cart button to unavailable class
$('add').disabled = true; // disable add-to-cart button $('add').disabled = true; // disable add-to-cart button
$('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message $('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message
} }
}; };
// initialize multi selector for product // initialize multi selector for product
window.addEvent('domready', function() { window.addEvent('domready', function() {
new Shopify.OptionSelectors("product-select", { product: {{ product | json }}, onVariantSelected: selectCallback }); new Shopify.OptionSelectors("product-select", { product: {{ product | json }}, onVariantSelected: selectCallback });
}); });
--> -->
</script> </script>

View File

@@ -1,85 +1,85 @@
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml" lang="en"> <html xmlns="http://www.w3.org/1999/xhtml" lang="en">
<head> <head>
<title>{{shop.name}} - {{page_title}}</title> <title>{{shop.name}} - {{page_title}}</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
{{ 'main.css' | asset_url | stylesheet_tag }} {{ 'main.css' | asset_url | stylesheet_tag }}
{{ 'shop.js' | asset_url | script_tag }} {{ 'shop.js' | asset_url | script_tag }}
{{ 'mootools.js' | asset_url | script_tag }} {{ 'mootools.js' | asset_url | script_tag }}
{{ 'slimbox.js' | asset_url | script_tag }} {{ 'slimbox.js' | asset_url | script_tag }}
{{ 'option_selection.js' | shopify_asset_url | script_tag }} {{ 'option_selection.js' | shopify_asset_url | script_tag }}
{{ 'slimbox.css' | asset_url | stylesheet_tag }} {{ 'slimbox.css' | asset_url | stylesheet_tag }}
{{ content_for_header }} {{ content_for_header }}
</head> </head>
<body id="page-{{template}}"> <body id="page-{{template}}">
<p class="hide"><a href="#navigation">Skip to navigation.</a></p> <p class="hide"><a href="#navigation">Skip to navigation.</a></p>
<div id="wrapper"> <div id="wrapper">
<div class="content clearfix"> <div class="content clearfix">
<div id="header"> <div id="header">
<h2><a href="/">{{shop.name}}</a></h2> <h2><a href="/">{{shop.name}}</a></h2>
</div> </div>
<div id="left-col"> <div id="left-col">
{{ content_for_layout }} {{ content_for_layout }}
</div> </div>
<div id="right-col"> <div id="right-col">
{% if template != 'cart' %} {% if template != 'cart' %}
<div id="cart-right-col"> <div id="cart-right-col">
<dl id="cart-right-col-info"> <dl id="cart-right-col-info">
<dt>Shopping Cart</dt> <dt>Shopping Cart</dt>
<dd> <dd>
{% if cart.item_count != 0 %} {% if cart.item_count != 0 %}
<a href="/cart">{{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }}</a> in your cart <a href="/cart">{{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }}</a> in your cart
{% else %} {% else %}
Your cart is empty Your cart is empty
{% endif %} {% endif %}
</dd> </dd>
</dl> </dl>
</div> </div>
{% endif %} {% endif %}
<div id="search"> <div id="search">
<dl id="searchbox"> <dl id="searchbox">
<dt>Search</dt> <dt>Search</dt>
<dd> <dd>
<form action="/search" method="get"> <form action="/search" method="get">
<fieldset> <fieldset>
<input class="search-input" type="text" onclick="this.select()" value="Search this shop..." name="q" /> <input class="search-input" type="text" onclick="this.select()" value="Search this shop..." name="q" />
</fieldset> </fieldset>
</form> </form>
</dd> </dd>
</dl> </dl>
</div> </div>
<div id="navigation"> <div id="navigation">
<dl class="navbar"> <dl class="navbar">
<dt>Navigation</dt> <dt>Navigation</dt>
{% for link in linklists.main-menu.links %} {% for link in linklists.main-menu.links %}
<dd>{{ link.title | link_to: link.url }}</dd> <dd>{{ link.title | link_to: link.url }}</dd>
{% endfor %} {% endfor %}
</dl> </dl>
{% if tags %} {% if tags %}
<dl class="navbar"> <dl class="navbar">
<dt>Tags</dt> <dt>Tags</dt>
{% for tag in collection.tags %} {% for tag in collection.tags %}
<dd>{{ tag | highlight_active_tag | link_to_tag: tag }}</dd> <dd>{{ tag | highlight_active_tag | link_to_tag: tag }}</dd>
{% endfor %} {% endfor %}
</dl> </dl>
{% endif %} {% endif %}
</div> </div>
</div> </div>
</div> </div>
<div id="content-padding"></div> <div id="content-padding"></div>
</div> </div>
<div id="footer"> <div id="footer">
{% for link in linklists.footer.links %} {% for link in linklists.footer.links %}
{{ link.title | link_to: link.url }} {% if forloop.rindex != 1 %} | {% endif %} {{ link.title | link_to: link.url }} {% if forloop.rindex != 1 %} | {% endif %}
{% endfor %} {% endfor %}
</div> </div>
</body> </body>
</html> </html>

View File

@@ -1,56 +1,56 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<div id="text-page">
<div class="entry">
<h1>Oh no!</h1>
<div class="entry-post">
Seems like you are looking for something that just isn't here. <a href="/">Try heading back to our main page</a>. Or you can checkout some of our featured products below.
</div>
</div>
</div>
<div id="text-page">
<h1>Featured Products</h1> <div class="entry">
<ul class="item-list clearfix"> <h1>Oh no!</h1>
<div class="entry-post">
{% for product in collections.frontpage.products %} Seems like you are looking for something that just isn't here. <a href="/">Try heading back to our main page</a>. Or you can checkout some of our featured products below.
<li> </div>
<form action="/cart/add" method="post"> </div>
<div class="item-list-item"> </div>
<div class="ili-top clearfix">
<div class="ili-top-content">
<h2><a href="{{product.url}}">{{product.title}}</a></h2>
<p>{{ product.description | truncatewords: 15 }}</p>
</div>
<a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</div>
<div class="ili-bottom clearfix">
<p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p>
<input type="submit" class="" value="Add to Basket" />
<p>
<a href="{{product.url}}">View Details</a>
<span>
{% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %}
{{product.compare_at_price | money}} -
{% endif %}
{% endif %}
<strong>
{{product.price_min | money}}
</strong>
</span>
</p>
</div>
</div>
</form>
</li>
{% endfor %}
</ul>
</div> <h1>Featured Products</h1>
<!-- end page --> <ul class="item-list clearfix">
{% for product in collections.frontpage.products %}
<li>
<form action="/cart/add" method="post">
<div class="item-list-item">
<div class="ili-top clearfix">
<div class="ili-top-content">
<h2><a href="{{product.url}}">{{product.title}}</a></h2>
<p>{{ product.description | truncatewords: 15 }}</p>
</div>
<a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</div>
<div class="ili-bottom clearfix">
<p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p>
<input type="submit" class="" value="Add to Basket" />
<p>
<a href="{{product.url}}">View Details</a>
<span>
{% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %}
{{product.compare_at_price | money}} -
{% endif %}
{% endif %}
<strong>
{{product.price_min | money}}
</strong>
</span>
</p>
</div>
</div>
</form>
</li>
{% endfor %}
</ul>
</div>
<!-- end page -->

View File

@@ -1,19 +1,19 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<div id="text-page"> <div id="text-page">
<div class="entry"> <div class="entry">
<h1><span>{{article.title}}</span></h1> <h1><span>{{article.title}}</span></h1>
<div class="entry-post"> <div class="entry-post">
<div class="meta">{{ article.created_at | date: "%b %d" }}</div> <div class="meta">{{ article.created_at | date: "%b %d" }}</div>
{{ article.content }} {{ article.content }}
</div> </div>
<!-- Comments --> <!-- Comments -->
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
<div id="comments"> <div id="comments">
<h2>Comments</h2> <h2>Comments</h2>
<!-- List all comments --> <!-- List all comments -->
<ul id="comment-list"> <ul id="comment-list">
{% for comment in article.comments %} {% for comment in article.comments %}
@@ -21,19 +21,19 @@
<div class="comment"> <div class="comment">
{{ comment.content }} {{ comment.content }}
</div> </div>
<div class="comment-details"> <div class="comment-details">
Posted by <span class="comment-author">{{ comment.author }}</span> on <span class="comment-date">{{ comment.created_at | date: "%B %d, %Y" }}</span> Posted by <span class="comment-author">{{ comment.author }}</span> on <span class="comment-date">{{ comment.created_at | date: "%B %d, %Y" }}</span>
</div> </div>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<!-- Comment Form --> <!-- Comment Form -->
<div id="comment-form"> <div id="comment-form">
{% form article %} {% form article %}
<h2>Leave a comment</h2> <h2>Leave a comment</h2>
<!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message --> <!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message -->
{% if form.posted_successfully? %} {% if form.posted_successfully? %}
{% if blog.moderated? %} {% if blog.moderated? %}
@@ -45,11 +45,11 @@
<div class="notice">Successfully posted your comment.</div> <div class="notice">Successfully posted your comment.</div>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% if form.errors %} {% if form.errors %}
<div class="notice error">Not all the fields have been filled out correctly!</div> <div class="notice error">Not all the fields have been filled out correctly!</div>
{% endif %} {% endif %}
<dl> <dl>
<dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt> <dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt>
<dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd> <dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd>
@@ -60,39 +60,39 @@
<dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt> <dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt>
<dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd> <dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd>
</dl> </dl>
{% if blog.moderated? %} {% if blog.moderated? %}
<p class="hint">comments have to be approved before showing up</p> <p class="hint">comments have to be approved before showing up</p>
{% endif %} {% endif %}
<input type="submit" value="Post comment" id="comment-submit" /> <input type="submit" value="Post comment" id="comment-submit" />
{% endform %} {% endform %}
</div> </div>
<!-- END Comment Form --> <!-- END Comment Form -->
</div> </div>
{% endif %} {% endif %}
<!-- END Comments --> <!-- END Comments -->
</div> </div>
</div> </div>
<div id="three-reasons" class="clearfix"> <div id="three-reasons" class="clearfix">
<h3>Why Shop With Us?</h3> <h3>Why Shop With Us?</h3>
<ul> <ul>
<li class="two-a"> <li class="two-a">
<h4>24 Hours</h4> <h4>24 Hours</h4>
<p>We're always here to help.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-c"> <li class="two-c">
<h4>No Spam</h4> <h4>No Spam</h4>
<p>We'll never share your info.</p> <p>We'll never share your info.</p>
</li> </li>
<li class="two-d"> <li class="two-d">
<h4>Secure Servers</h4> <h4>Secure Servers</h4>
<p>Checkout is 256bit encrypted.</p> <p>Checkout is 256bit encrypted.</p>
</li> </li>
</ul> </ul>
</div> </div>
</div> </div>

View File

@@ -1,41 +1,41 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<div id="text-page"> <div id="text-page">
<h1>Post from our blog...</h1> <h1>Post from our blog...</h1>
{% paginate blog.articles by 20 %} {% paginate blog.articles by 20 %}
{% for article in blog.articles %} {% for article in blog.articles %}
<div class="entry">
<h1><span><a href="{{ article.url }}">{{ article.title }}</a></span></h1>
<div class="entry-post">
<div class="meta">{{ article.created_at | date: "%b %d" }}</div>
{{ article.content }}
</div>
</div>
{% endfor %} <div class="entry">
<h1><span><a href="{{ article.url }}">{{ article.title }}</a></span></h1>
<div class="entry-post">
<div class="meta">{{ article.created_at | date: "%b %d" }}</div>
{{ article.content }}
</div>
</div>
<div class="paginate clearfix"> {% endfor %}
{{ paginate | default_pagination }}
</div>
{% endpaginate %} <div class="paginate clearfix">
</div> {{ paginate | default_pagination }}
</div>
<div id="three-reasons" class="clearfix">
<h3>Why Shop With Us?</h3> {% endpaginate %}
<ul> </div>
<li class="two-a">
<h4>24 Hours</h4> <div id="three-reasons" class="clearfix">
<p>We're always here to help.</p> <h3>Why Shop With Us?</h3>
</li> <ul>
<li class="two-c"> <li class="two-a">
<h4>No Spam</h4> <h4>24 Hours</h4>
<p>We'll never share your info.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-d"> <li class="two-c">
<h4>Secure Servers</h4> <h4>No Spam</h4>
<p>Checkout is 256bit encrypted.</p> <p>We'll never share your info.</p>
</li> </li>
</ul> <li class="two-d">
</div> <h4>Secure Servers</h4>
</div> <p>Checkout is 256bit encrypted.</p>
</li>
</ul>
</div>
</div>

View File

@@ -1,134 +1,134 @@
<script type="text/javascript"> <script type="text/javascript">
function remove_item(id) { function remove_item(id) {
document.getElementById('updates_'+id).value = 0; document.getElementById('updates_'+id).value = 0;
document.getElementById('cart').submit(); document.getElementById('cart').submit();
} }
</script> </script>
<div id="page" class="innerpage clearfix">. <div id="page" class="innerpage clearfix">.
{% if cart.item_count == 0 %} {% if cart.item_count == 0 %}
<h1>Your cart is currently empty.</h1> <h1>Your cart is currently empty.</h1>
{% else %} {% else %}
<h1>Your Cart <span>({{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }}, {{cart.total_price | money_with_currency }} total)</span></h1> <h1>Your Cart <span>({{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }}, {{cart.total_price | money_with_currency }} total)</span></h1>
<form action="/cart" method="post" id="cart-form"> <form action="/cart" method="post" id="cart-form">
<div id="cart-wrap"> <div id="cart-wrap">
<table width="100%" border="0" cellspacing="0" cellpadding="0"> <table width="100%" border="0" cellspacing="0" cellpadding="0">
<tr> <tr>
<th scope="col" class="td-image"><label>Image</label></th> <th scope="col" class="td-image"><label>Image</label></th>
<th scope="col" class="td-title"><label>Product Title</label></th> <th scope="col" class="td-title"><label>Product Title</label></th>
<th scope="col" class="td-count"><label>Count</label></th> <th scope="col" class="td-count"><label>Count</label></th>
<th scope="col" class="td-price"><label>Cost</label></th> <th scope="col" class="td-price"><label>Cost</label></th>
<th scope="col" class="td-delete"><label>Remove</label></th> <th scope="col" class="td-delete"><label>Remove</label></th>
</tr> </tr>
{% for item in cart.items %} {% for item in cart.items %}
<tr class="{% cycle 'reg', 'alt' %}"> <tr class="{% cycle 'reg', 'alt' %}">
<td colspan="5"> <td colspan="5">
<table width="100%" border="0" cellspacing="0" cellpadding="0"> <table width="100%" border="0" cellspacing="0" cellpadding="0">
<tr> <tr>
<td class="td-image"><a href="{{item.product.url}}">{{ item.product.featured_image | product_img_url: 'thumb' | img_tag }}</a></td> <td class="td-image"><a href="{{item.product.url}}">{{ item.product.featured_image | product_img_url: 'thumb' | img_tag }}</a></td>
<td class="td-title"><p>{{ item.title }}</p></td> <td class="td-title"><p>{{ item.title }}</p></td>
<td class="td-count"><label>Count:</label> <input type="text" class="quantity item-count" name="updates[{{item.variant.id}}]" id="updates_{{item.variant.id}}" value="{{item.quantity}}" onfocus="this.select();"/></td> <td class="td-count"><label>Count:</label> <input type="text" class="quantity item-count" name="updates[{{item.variant.id}}]" id="updates_{{item.variant.id}}" value="{{item.quantity}}" onfocus="this.select();"/></td>
<td class="td-price">{{item.line_price | money }}</td> <td class="td-price">{{item.line_price | money }}</td>
<td class="td-delete"><a href="#" onclick="remove_item({{item.variant.id}}); return false;">Remove</a></td> <td class="td-delete"><a href="#" onclick="remove_item({{item.variant.id}}); return false;">Remove</a></td>
</tr> </tr>
</table> </table>
</td> </td>
</tr> </tr>
{% endfor %} {% endfor %}
</table> </table>
<div id="finish-up"> <div id="finish-up">
<div class="latest-news-box"> <div class="latest-news-box">
{{ pages.shopping-cart.content }} {{ pages.shopping-cart.content }}
</div> </div>
<p class="order-total"> <p class="order-total">
<span><strong>Order Total:</strong> {{cart.total_price | money_with_currency }}</span> <span><strong>Order Total:</strong> {{cart.total_price | money_with_currency }}</span>
</p> </p>
<p class="update-cart"><input type="submit" value="Refresh Cart" name="update" /></p> <p class="update-cart"><input type="submit" value="Refresh Cart" name="update" /></p>
<p class="go-checkout"><input type="submit" value="Proceed to Checkout" name="checkout" /></p> <p class="go-checkout"><input type="submit" value="Proceed to Checkout" name="checkout" /></p>
{% if additional_checkout_buttons %} {% if additional_checkout_buttons %}
<div class="additional-checkout-buttons"> <div class="additional-checkout-buttons">
<p>- or -</p> <p>- or -</p>
{{ content_for_additional_checkout_buttons }} {{ content_for_additional_checkout_buttons }}
</div> </div>
{% endif %} {% endif %}
</div> </div>
</div> </div>
</form> </form>
{% endif %} {% endif %}
<h1 class="other-products"><span>Other Products You Might Enjoy</span></h1> <h1 class="other-products"><span>Other Products You Might Enjoy</span></h1>
<ul class="item-list clearfix"> <ul class="item-list clearfix">
{% for product in collections.frontpage.products limit:2 %} {% for product in collections.frontpage.products limit:2 %}
<li> <li>
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<div class="item-list-item"> <div class="item-list-item">
<div class="ili-top clearfix"> <div class="ili-top clearfix">
<div class="ili-top-content"> <div class="ili-top-content">
<h2><a href="{{product.url}}">{{product.title}}</a></h2> <h2><a href="{{product.url}}">{{product.title}}</a></h2>
<p>{{ product.description | truncatewords: 15 }}</p> <p>{{ product.description | truncatewords: 15 }}</p>
</div> </div>
<a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a> <a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</div> </div>
<div class="ili-bottom clearfix"> <div class="ili-bottom clearfix">
<p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p> <p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p>
<input type="submit" class="" value="Add to Basket" /> <input type="submit" class="" value="Add to Basket" />
<p> <p>
<a href="{{product.url}}">View Details</a> <a href="{{product.url}}">View Details</a>
<span> <span>
{% if product.compare_at_price %} {% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %} {% if product.price_min != product.compare_at_price %}
{{product.compare_at_price | money}} - {{product.compare_at_price | money}} -
{% endif %} {% endif %}
{% endif %} {% endif %}
<strong> <strong>
{{product.price_min | money}} {{product.price_min | money}}
</strong> </strong>
</span> </span>
</p> </p>
</div> </div>
</div> </div>
</form> </form>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<div id="three-reasons" class="clearfix"> <div id="three-reasons" class="clearfix">
<h3>Why Shop With Us?</h3> <h3>Why Shop With Us?</h3>
<ul> <ul>
<li class="two-a"> <li class="two-a">
<h4>24 Hours</h4> <h4>24 Hours</h4>
<p>We're always here to help.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-c"> <li class="two-c">
<h4>No Spam</h4> <h4>No Spam</h4>
<p>We'll never share your info.</p> <p>We'll never share your info.</p>
</li> </li>
<li class="two-d"> <li class="two-d">
<h4>Secure Servers</h4> <h4>Secure Servers</h4>
<p>Checkout is 256bit encrypted.</p> <p>Checkout is 256bit encrypted.</p>
</li> </li>
</ul> </ul>
</div> </div>
</div> </div>
<!-- end page --> <!-- end page -->

View File

@@ -1,70 +1,70 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<h1>{{ collection.title }}</h1> <h1>{{ collection.title }}</h1>
{% if collection.description.size > 0 %} {% if collection.description.size > 0 %}
<div class="latest-news">{{ collection.description }}</div> <div class="latest-news">{{ collection.description }}</div>
{% endif %} {% endif %}
{% paginate collection.products by 8 %} {% paginate collection.products by 8 %}
<ul class="item-list clearfix"> <ul class="item-list clearfix">
{% for product in collection.products %} {% for product in collection.products %}
<li> <li>
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<div class="item-list-item"> <div class="item-list-item">
<div class="ili-top clearfix"> <div class="ili-top clearfix">
<div class="ili-top-content"> <div class="ili-top-content">
<h2><a href="{{product.url}}">{{product.title}}</a></h2> <h2><a href="{{product.url}}">{{product.title}}</a></h2>
<p>{{ product.description | truncatewords: 15 }}</p> <p>{{ product.description | truncatewords: 15 }}</p>
</div> </div>
<a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a> <a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</div> </div>
<div class="ili-bottom clearfix"> <div class="ili-bottom clearfix">
<p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p> <p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p>
<input type="submit" class="" value="Add to Basket" /> <input type="submit" class="" value="Add to Basket" />
<p> <p>
<a href="{{product.url}}">View Details</a> <a href="{{product.url}}">View Details</a>
<span> <span>
{% if product.compare_at_price %} {% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %} {% if product.price_min != product.compare_at_price %}
{{product.compare_at_price | money}} - {{product.compare_at_price | money}} -
{% endif %} {% endif %}
{% endif %} {% endif %}
<strong> <strong>
{{product.price_min | money}} {{product.price_min | money}}
</strong> </strong>
</span> </span>
</p> </p>
</div> </div>
</div> </div>
</form> </form>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<div class="paginate clearfix"> <div class="paginate clearfix">
{{ paginate | default_pagination }} {{ paginate | default_pagination }}
</div> </div>
<div id="three-reasons" class="clearfix"> <div id="three-reasons" class="clearfix">
<h3>Why Shop With Us?</h3> <h3>Why Shop With Us?</h3>
<ul> <ul>
<li class="two-a"> <li class="two-a">
<h4>24 Hours</h4> <h4>24 Hours</h4>
<p>We're always here to help.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-c"> <li class="two-c">
<h4>No Spam</h4> <h4>No Spam</h4>
<p>We'll never share your info.</p> <p>We'll never share your info.</p>
</li> </li>
<li class="two-d"> <li class="two-d">
<h4>Secure Servers</h4> <h4>Secure Servers</h4>
<p>Checkout is 256bit encrypted.</p> <p>Checkout is 256bit encrypted.</p>
</li> </li>
</ul> </ul>
</div> </div>
</div> </div>
{% endpaginate %} {% endpaginate %}

View File

@@ -1,94 +1,94 @@
<div id="gwrap"> <div id="gwrap">
<div id="gbox"> <div id="gbox">
<h1>Three Great Reasons You Should Shop With Us...</h1> <h1>Three Great Reasons You Should Shop With Us...</h1>
<ul> <ul>
<li class="gbox1"> <li class="gbox1">
<h2>Free Shipping</h2> <h2>Free Shipping</h2>
<p>On all orders over $25</p> <p>On all orders over $25</p>
</li> </li>
<li class="gbox2"> <li class="gbox2">
<h2>Top Quality</h2> <h2>Top Quality</h2>
<p>Hand made in our shop</p> <p>Hand made in our shop</p>
</li> </li>
<li class="gbox3"> <li class="gbox3">
<h2>100% Guarantee</h2> <h2>100% Guarantee</h2>
<p>Any time, any reason</p> <p>Any time, any reason</p>
</li> </li>
</ul> </ul>
</div> </div>
</div> </div>
<div id="page" class="clearfix"> <div id="page" class="clearfix">
<div class="latest-news">{{pages.alert.content}}</div> <div class="latest-news">{{pages.alert.content}}</div>
<ul class="item-list clearfix"> <ul class="item-list clearfix">
{% for product in collections.frontpage.products %} {% for product in collections.frontpage.products %}
<li> <li>
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<div class="item-list-item"> <div class="item-list-item">
<div class="ili-top clearfix"> <div class="ili-top clearfix">
<div class="ili-top-content"> <div class="ili-top-content">
<h2><a href="{{product.url}}">{{product.title}}</a></h2> <h2><a href="{{product.url}}">{{product.title}}</a></h2>
{{ product.description | truncatewords: 15 }}</p> <!-- extra cloding <p> tag for truncation --> {{ product.description | truncatewords: 15 }}</p> <!-- extra cloding <p> tag for truncation -->
</div> </div>
<a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a> <a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</div> </div>
<div class="ili-bottom clearfix"> <div class="ili-bottom clearfix">
<p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p> <p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p>
<input type="submit" class="" value="Add to Basket" /> <input type="submit" class="" value="Add to Basket" />
<p> <p>
<a href="{{product.url}}">View Details</a> <a href="{{product.url}}">View Details</a>
<span> <span>
{% if product.compare_at_price %} {% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %} {% if product.price_min != product.compare_at_price %}
{{product.compare_at_price | money}} - {{product.compare_at_price | money}} -
{% endif %} {% endif %}
{% endif %} {% endif %}
<strong> <strong>
{{product.price_min | money}} {{product.price_min | money}}
</strong> </strong>
</span> </span>
</p> </p>
</div> </div>
</div> </div>
</form> </form>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<div id="one-two"> <div id="one-two">
<div id="two"> <div id="two">
<h3>Why Shop With Us?</h3> <h3>Why Shop With Us?</h3>
<ul> <ul>
<li class="two-a"> <li class="two-a">
<h4>24 Hours</h4> <h4>24 Hours</h4>
<p>We're always here to help.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-c"> <li class="two-c">
<h4>No Spam</h4> <h4>No Spam</h4>
<p>We'll never share your info.</p> <p>We'll never share your info.</p>
</li> </li>
<li class="two-b"> <li class="two-b">
<h4>Save Energy</h4> <h4>Save Energy</h4>
<p>We're green, all the way.</p> <p>We're green, all the way.</p>
</li> </li>
<li class="two-d"> <li class="two-d">
<h4>Secure Servers</h4> <h4>Secure Servers</h4>
<p>Checkout is 256bits encrypted.</p> <p>Checkout is 256bits encrypted.</p>
</li> </li>
</ul> </ul>
</div> </div>
<div id="one"> <div id="one">
<h3>Our Company</h3> <h3>Our Company</h3>
{{pages.about-us.content | truncatewords: 49}} <a href="/pages/about-us">read more</a></p> {{pages.about-us.content | truncatewords: 49}} <a href="/pages/about-us">read more</a></p>
</div> </div>
</div> </div>
</div> </div>
<!-- end page --> <!-- end page -->

View File

@@ -1,56 +1,56 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<div id="text-page"> <div id="text-page">
<div class="entry"> <div class="entry">
<h1>{{page.title}}</h1> <h1>{{page.title}}</h1>
<div class="entry-post"> <div class="entry-post">
{{page.content}} {{page.content}}
</div> </div>
</div> </div>
</div> </div>
<h1>Featured Products</h1> <h1>Featured Products</h1>
<ul class="item-list clearfix"> <ul class="item-list clearfix">
{% for product in collections.frontpage.products %} {% for product in collections.frontpage.products %}
<li> <li>
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<div class="item-list-item"> <div class="item-list-item">
<div class="ili-top clearfix"> <div class="ili-top clearfix">
<div class="ili-top-content"> <div class="ili-top-content">
<h2><a href="{{product.url}}">{{product.title}}</a></h2> <h2><a href="{{product.url}}">{{product.title}}</a></h2>
<p>{{ product.description | truncatewords: 15 }}</p> <p>{{ product.description | truncatewords: 15 }}</p>
</div> </div>
<a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a> <a href="{{product.url}}" class="ili-top-image"><img src="{{ product.featured_image | product_img_url: 'small' }}" alt="{{ product.title | escape }}"/></a>
</div> </div>
<div class="ili-bottom clearfix"> <div class="ili-bottom clearfix">
<p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p> <p class="hiddenvariants" style="display: none">{% for variant in product.variants %}<span><input type="radio" name="id" value="{{variant.id}}" id="radio_{{variant.id}}" style="vertical-align: middle;" {%if forloop.first%} checked="checked" {%endif%} /><label for="radio_{{variant.id}}">{{ variant.price | money_with_currency }} - {{ variant.title }}</label></span>{% endfor %}</p>
<input type="submit" class="" value="Add to Basket" /> <input type="submit" class="" value="Add to Basket" />
<p> <p>
<a href="{{product.url}}">View Details</a> <a href="{{product.url}}">View Details</a>
<span> <span>
{% if product.compare_at_price %} {% if product.compare_at_price %}
{% if product.price_min != product.compare_at_price %} {% if product.price_min != product.compare_at_price %}
{{product.compare_at_price | money}} - {{product.compare_at_price | money}} -
{% endif %} {% endif %}
{% endif %} {% endif %}
<strong> <strong>
{{product.price_min | money}} {{product.price_min | money}}
</strong> </strong>
</span> </span>
</p> </p>
</div> </div>
</div> </div>
</form> </form>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
</div> </div>
<!-- end page --> <!-- end page -->

View File

@@ -1,116 +1,116 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<h1>{{ collection.title }} {{ product.title }}</h1> <h1>{{ collection.title }} {{ product.title }}</h1>
<p class="latest-news"><strong>Product Tags: </strong> <p class="latest-news"><strong>Product Tags: </strong>
{% for tag in product.tags %} {% for tag in product.tags %}
<a href="/collections/all/{{ tag }}">{{ tag }}</a> | <a href="/collections/all/{{ tag }}">{{ tag }}</a> |
{% endfor %} {% endfor %}
</p> </p>
<div class="product clearfix"> <div class="product clearfix">
<div class="product-info"> <div class="product-info">
<h1>{{ product.title }}</h1> <h1>{{ product.title }}</h1>
<div class="product-info-description"> <div class="product-info-description">
<p>{{ product.description }} </p> <p>{{ product.description }} </p>
</div> </div>
{% if product.available %} {% if product.available %}
<form action="/cart/add" method="post"> <form action="/cart/add" method="post">
<h2>Product Options:</h2> <h2>Product Options:</h2>
<select id="product-info-options" name="id" class="product-info-options"> <select id="product-info-options" name="id" class="product-info-options">
{% for variant in product.variants %} {% for variant in product.variants %}
<option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option> <option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option>
{% endfor %} {% endfor %}
</select> </select>
<div id="price-field"></div> <div id="price-field"></div>
<div class="product-purchase-btn"> <div class="product-purchase-btn">
<input type="submit" class="add-this-to-cart" id="add-this-to-cart" value="Add to Basket" /> <input type="submit" class="add-this-to-cart" id="add-this-to-cart" value="Add to Basket" />
</div> </div>
</form> </form>
{% else %} {% else %}
<h2>Sold out!</h2> <h2>Sold out!</h2>
<p>Sorry, we're all out of this product. Check back often and order when it returns</p> <p>Sorry, we're all out of this product. Check back often and order when it returns</p>
{% endif %} {% endif %}
</div> </div>
<div class="product-images clearfix"> <div class="product-images clearfix">
{% for image in product.images %} {% for image in product.images %}
{% if forloop.first %} {% if forloop.first %}
<div class="product-image-large"> <div class="product-image-large">
<img src="{{ image | product_img_url: 'medium'}}" alt="{{product.title | escape }}" /> <img src="{{ image | product_img_url: 'medium'}}" alt="{{product.title | escape }}" />
</div> </div>
{% else %} {% else %}
{% endif %} {% endif %}
{% endfor %} {% endfor %}
<ul class="product-thumbs clearfix"> <ul class="product-thumbs clearfix">
{% for image in product.images %} {% for image in product.images %}
{% if forloop.first %} {% if forloop.first %}
{% else %} {% else %}
<li> <li>
<a href="{{ image | product_img_url: 'large' }}" class="product-thumbs" rel="lightbox[product]" title=""> <a href="{{ image | product_img_url: 'large' }}" class="product-thumbs" rel="lightbox[product]" title="">
<img src="{{ image | product_img_url: 'small'}}" alt="{{product.title | escape }}" /> <img src="{{ image | product_img_url: 'small'}}" alt="{{product.title | escape }}" />
</a> </a>
</li> </li>
{% endif %} {% endif %}
{% endfor %} {% endfor %}
</ul> </ul>
</div> </div>
</div> </div>
<div id="three-reasons" class="clearfix"> <div id="three-reasons" class="clearfix">
<h3>Why Shop With Us?</h3> <h3>Why Shop With Us?</h3>
<ul> <ul>
<li class="two-a"> <li class="two-a">
<h4>24 Hours</h4> <h4>24 Hours</h4>
<p>We're always here to help.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-c"> <li class="two-c">
<h4>No Spam</h4> <h4>No Spam</h4>
<p>We'll never share your info.</p> <p>We'll never share your info.</p>
</li> </li>
<li class="two-d"> <li class="two-d">
<h4>Secure Servers</h4> <h4>Secure Servers</h4>
<p>Checkout is 256bit encrypted.</p> <p>Checkout is 256bit encrypted.</p>
</li> </li>
</ul> </ul>
</div> </div>
</div> </div>
<!-- end page --> <!-- end page -->
<script type="text/javascript"> <script type="text/javascript">
<!-- <!--
// prototype callback for multi variants dropdown selector // prototype callback for multi variants dropdown selector
var selectCallback = function(variant, selector) { var selectCallback = function(variant, selector) {
if (variant && variant.available == true) { if (variant && variant.available == true) {
// selected a valid variant // selected a valid variant
$('add-this-to-cart').removeClassName('disabled'); // remove unavailable class from add-to-cart button $('add-this-to-cart').removeClassName('disabled'); // remove unavailable class from add-to-cart button
$('add-this-to-cart').disabled = false; // reenable add-to-cart button $('add-this-to-cart').disabled = false; // reenable add-to-cart button
$('price-field').innerHTML = Shopify.formatMoney(variant.price, "{{shop.money_with_currency_format}}"); // update price field $('price-field').innerHTML = Shopify.formatMoney(variant.price, "{{shop.money_with_currency_format}}"); // update price field
} else { } else {
// variant doesn't exist // variant doesn't exist
$('add-this-to-cart').addClassName('disabled'); // set add-to-cart button to unavailable class $('add-this-to-cart').addClassName('disabled'); // set add-to-cart button to unavailable class
$('add-this-to-cart').disabled = true; // disable add-to-cart button $('add-this-to-cart').disabled = true; // disable add-to-cart button
$('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message $('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message
} }
}; };
// initialize multi selector for product // initialize multi selector for product
Event.observe(document, 'dom:loaded', function() { Event.observe(document, 'dom:loaded', function() {
new Shopify.OptionSelectors("product-info-options", { product: {{ product | json }}, onVariantSelected: selectCallback }); new Shopify.OptionSelectors("product-info-options", { product: {{ product | json }}, onVariantSelected: selectCallback });
}); });
--> -->
</script> </script>

View File

@@ -2,50 +2,50 @@
<div id="page" class="innerpage clearfix"> <div id="page" class="innerpage clearfix">
<h1>Search Results</h1> <h1>Search Results</h1>
{% if search.performed %} {% if search.performed %}
{% paginate search.results by 10 %} {% paginate search.results by 10 %}
{% if search.results == empty %} {% if search.results == empty %}
<div class="latest-news">Your search for "{{search.terms | escape}}" did not yield any results</div> <div class="latest-news">Your search for "{{search.terms | escape}}" did not yield any results</div>
{% else %} {% else %}
<ul class="search-list clearfix"> <ul class="search-list clearfix">
{% for item in search.results %} {% for item in search.results %}
<li> <li>
<h3 class="stitle">{{ item.title | link_to: item.url }}</h3> <h3 class="stitle">{{ item.title | link_to: item.url }}</h3>
<p class="sinfo">{{ item.content | strip_html | truncatewords: 65 | highlight: search.terms }} ... <a href="{{item.url}}" title="">view this item</a></p> <p class="sinfo">{{ item.content | strip_html | truncatewords: 65 | highlight: search.terms }} ... <a href="{{item.url}}" title="">view this item</a></p>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
{% endif %} {% endif %}
<div class="paginate clearfix">
{{ paginate | default_pagination }}
</div>
<div class="paginate clearfix">
<div id="three-reasons" class="clearfix"> {{ paginate | default_pagination }}
<h3>Why Shop With Us?</h3> </div>
<ul>
<li class="two-a">
<h4>24 Hours</h4> <div id="three-reasons" class="clearfix">
<p>We're always here to help.</p> <h3>Why Shop With Us?</h3>
</li> <ul>
<li class="two-c"> <li class="two-a">
<h4>No Spam</h4> <h4>24 Hours</h4>
<p>We'll never share your info.</p> <p>We're always here to help.</p>
</li> </li>
<li class="two-d"> <li class="two-c">
<h4>Secure Servers</h4> <h4>No Spam</h4>
<p>Checkout is 256bit encrypted.</p> <p>We'll never share your info.</p>
</li> </li>
</ul> <li class="two-d">
</div> <h4>Secure Servers</h4>
</div> <p>Checkout is 256bit encrypted.</p>
</li>
</ul>
</div>
</div>
{% endpaginate %} {% endpaginate %}
{% endif %} {% endif %}

View File

@@ -2,15 +2,15 @@
<html xmlns="http://www.w3.org/1999/xhtml"> <html xmlns="http://www.w3.org/1999/xhtml">
<head> <head>
<title>{{shop.name}} - {{page_title}}</title> <title>{{shop.name}} - {{page_title}}</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> <meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
{{ 'reset.css' | asset_url | stylesheet_tag }} {{ 'reset.css' | asset_url | stylesheet_tag }}
{{ 'style.css' | asset_url | stylesheet_tag }} {{ 'style.css' | asset_url | stylesheet_tag }}
{{ 'lightbox.css' | asset_url | stylesheet_tag }} {{ 'lightbox.css' | asset_url | stylesheet_tag }}
{{ 'prototype/1.6/prototype.js' | global_asset_url | script_tag }} {{ 'prototype/1.6/prototype.js' | global_asset_url | script_tag }}
{{ 'scriptaculous/1.8.2/scriptaculous.js' | global_asset_url | script_tag }} {{ 'scriptaculous/1.8.2/scriptaculous.js' | global_asset_url | script_tag }}
{{ 'lightbox.js' | asset_url | script_tag }} {{ 'lightbox.js' | asset_url | script_tag }}
{{ 'option_selection.js' | shopify_asset_url | script_tag }} {{ 'option_selection.js' | shopify_asset_url | script_tag }}
{{ content_for_header }} {{ content_for_header }}
@@ -18,73 +18,73 @@
<body id="page-{{template}}"> <body id="page-{{template}}">
<div id="wrap"> <div id="wrap">
<div id="top">
<div id="cart">
<h3>Shopping Cart</h3>
<p class="cart-count">
{% if cart.item_count == 0 %}
Your cart is currently empty
{% else %}
{{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }} <span>-</span> Total: {{cart.total_price | money_with_currency }} <span>-</span> <a href="/cart">View Cart</a>
{% endif %}
</p>
</div>
<div id="site-title">
<h3><a href="/">{{shop.name}}</a></h3>
<h4><span>Tribble: A Shopify Theme</span></h4>
</div>
</div>
<ul id="nav">
{% for link in linklists.main-menu.links %}
<li>{{ link.title | link_to: link.url }}</li>
{% endfor %}
</ul>
{{ content_for_layout }}
<div id="foot" class="clearfix"> <div id="top">
<div class="quick-links"> <div id="cart">
<h4>Quick Navigation</h4> <h3>Shopping Cart</h3>
<ul class="clearfix"> <p class="cart-count">
<li><a href="/">Home</a></li> {% if cart.item_count == 0 %}
<li><a href="#top">Back to top</a></li> Your cart is currently empty
{% for link in linklists.main-menu.links %} {% else %}
<li>{{ link.title | link_to: link.url }}</li> {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }} <span>-</span> Total: {{cart.total_price | money_with_currency }} <span>-</span> <a href="/cart">View Cart</a>
{% endfor %} {% endif %}
</ul> </p>
</div> </div>
<div class="quick-contact"> <div id="site-title">
<h4>Quick Contact</h4> <h3><a href="/">{{shop.name}}</a></h3>
<div class="vcard"> <h4><span>Tribble: A Shopify Theme</span></h4>
<div class="org fn"> </div>
<div class="organization-name">Really Great Widget Co.</div> </div>
</div>
<div class="adr"> <ul id="nav">
<span class="street-address">2531 Barrington Court</span> {% for link in linklists.main-menu.links %}
<span class="locality">Hayward</span>, <li>{{ link.title | link_to: link.url }}</li>
<abbr title="California" class="region">CA</abbr> {% endfor %}
<span class="postal-code">94545</span> </ul>
</div>
<a class="email" href="mailto:email@myshopifysite.com"> {{ content_for_layout }}
email@myshopifysite.com
</a> <div id="foot" class="clearfix">
<div class="tel"> <div class="quick-links">
<span class="type">Support:</span> <span class="value">800-555-9954</span> <h4>Quick Navigation</h4>
</div> <ul class="clearfix">
</div> <li><a href="/">Home</a></li>
<li><a href="#top">Back to top</a></li>
{% for link in linklists.main-menu.links %}
<li>{{ link.title | link_to: link.url }}</li>
{% endfor %}
</ul>
</div>
<div class="quick-contact">
<h4>Quick Contact</h4>
<div class="vcard">
<div class="org fn">
<div class="organization-name">Really Great Widget Co.</div>
</div>
<div class="adr">
<span class="street-address">2531 Barrington Court</span>
<span class="locality">Hayward</span>,
<abbr title="California" class="region">CA</abbr>
<span class="postal-code">94545</span>
</div>
<a class="email" href="mailto:email@myshopifysite.com">
email@myshopifysite.com
</a>
<div class="tel">
<span class="type">Support:</span> <span class="value">800-555-9954</span>
</div>
</div>
</div>
<p><a href="http://shopify.com" class="we-made">Powered by Shopify</a> &copy; Copyright {{ "now" | date: "%Y" }} {{ shop.name }}, All Rights Reserved. <a href="/blogs/news.xml" id="foot-rss">RSS Feed</a></p>
</div>
</div>
<p><a href="http://shopify.com" class="we-made">Powered by Shopify</a> &copy; Copyright {{ "now" | date: "%Y" }} {{ shop.name }}, All Rights Reserved. <a href="/blogs/news.xml" id="foot-rss">RSS Feed</a></p>
</div>
</div> </div>
</body> </body>
</html> </html>

View File

@@ -3,7 +3,7 @@
<p> posted {{ article.created_at | date: "%Y %h" }} by {{ article.author }}</p> <p> posted {{ article.created_at | date: "%Y %h" }} by {{ article.author }}</p>
<div class="article-body textile"> <div class="article-body textile">
{{ article.content }} {{ article.content }}
</div> </div>
</div> </div>
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
@@ -17,18 +17,18 @@
<div class="comment"> <div class="comment">
{{ comment.content }} {{ comment.content }}
</div> </div>
<div class="comment-details"> <div class="comment-details">
Posted by {{ comment.author }} on {{ comment.created_at | date: "%B %d, %Y" }} Posted by {{ comment.author }} on {{ comment.created_at | date: "%B %d, %Y" }}
</div> </div>
</li> </li>
{% endfor %} {% endfor %}
</ul> </ul>
<!-- Comment Form --> <!-- Comment Form -->
{% form article %} {% form article %}
<h3>Leave a comment</h3> <h3>Leave a comment</h3>
<!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message --> <!-- Check if a comment has been submitted in the last request, and if yes display an appropriate message -->
{% if form.posted_successfully? %} {% if form.posted_successfully? %}
{% if blog.moderated? %} {% if blog.moderated? %}
@@ -40,11 +40,11 @@
<div class="notice">Successfully posted your comment.</div> <div class="notice">Successfully posted your comment.</div>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% if form.errors %} {% if form.errors %}
<div class="notice error">Not all the fields have been filled out correctly!</div> <div class="notice error">Not all the fields have been filled out correctly!</div>
{% endif %} {% endif %}
<dl> <dl>
<dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt> <dt class="{% if form.errors contains 'author' %}error{% endif %}"><label for="comment_author">Your name</label></dt>
<dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd> <dd><input type="text" id="comment_author" name="comment[author]" size="40" value="{{form.author}}" class="{% if form.errors contains 'author' %}input-error{% endif %}" /></dd>
@@ -55,12 +55,12 @@
<dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt> <dt class="{% if form.errors contains 'body' %}error{% endif %}"><label for="comment_body">Your comment</label></dt>
<dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd> <dd><textarea id="comment_body" name="comment[body]" cols="40" rows="5" class="{% if form.errors contains 'body' %}input-error{% endif %}">{{form.body}}</textarea></dd>
</dl> </dl>
{% if blog.moderated? %} {% if blog.moderated? %}
<p class="hint">comments have to be approved before showing up</p> <p class="hint">comments have to be approved before showing up</p>
{% endif %} {% endif %}
<input type="submit" value="Post comment" /> <input type="submit" value="Post comment" />
{% endform %} {% endform %}
</div> </div>
{% endif %} {% endif %}

View File

@@ -8,16 +8,16 @@
<h3 class="article-head-title"> <h3 class="article-head-title">
<a href="{{article.url}}">{{ article.title }}</a> <a href="{{article.url}}">{{ article.title }}</a>
</h3> </h3>
<p> <p>
{% if blog.comments_enabled? %} {% if blog.comments_enabled? %}
<a href="{{article.url}}#comments">{{ article.comments_count }} comments</a> <a href="{{article.url}}#comments">{{ article.comments_count }} comments</a>
&mdash; &mdash;
{% endif %} {% endif %}
posted {{ article.created_at | date: "%Y %h" }} by {{ article.author }}</p> posted {{ article.created_at | date: "%Y %h" }} by {{ article.author }}</p>
<div class="article-body textile"> <div class="article-body textile">
{{ article.content }} {{ article.content }}
</div> </div>
</div> </div>
{% endfor %} {% endfor %}
@@ -29,4 +29,4 @@
</div> </div>
<div class="clear-me"></div> <div class="clear-me"></div>

View File

@@ -1,17 +1,17 @@
<h1>Shopping Cart</h1> <h1>Shopping Cart</h1>
{% if cart.item_count == 0 %} {% if cart.item_count == 0 %}
<p><strong>Your shopping basket is empty.</strong> Perhaps a featured item below is of interest...</p> <p><strong>Your shopping basket is empty.</strong> Perhaps a featured item below is of interest...</p>
<table id="gallery"> <table id="gallery">
{% tablerow product in collections.frontpage.products cols: 3 limit: 12 %} {% tablerow product in collections.frontpage.products cols: 3 limit: 12 %}
<div class="gallery-image"> <div class="gallery-image">
<a href="{{ product.url | within: collections.frontpage }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 | escape }}"><img src="{{ product.images.first | product_img_url: 'medium' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ product.url | within: collections.frontpage }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 | escape }}"><img src="{{ product.images.first | product_img_url: 'medium' }}" alt="{{ product.title | escape }}" /></a>
</div> </div>
<div class="gallery-info"> <div class="gallery-info">
<a href="{{ product.url | within: collections.frontpage }}">{{ product.title | truncate: 30 }}</a><br /> <a href="{{ product.url | within: collections.frontpage }}">{{ product.title | truncate: 30 }}</a><br />
<small>{{ product.price | money }}{% if product.compare_at_price_max > product.price %} <del>{{ product.compare_at_price_max | money }}</del>{% endif %}</small> <small>{{ product.price | money }}{% if product.compare_at_price_max > product.price %} <del>{{ product.compare_at_price_max | money }}</del>{% endif %}</small>
</div> </div>
{% endtablerow %} {% endtablerow %}
</table> </table>
{% else %} {% else %}
<script type="text/javascript"> <script type="text/javascript">
function remove_item(id) { function remove_item(id) {
@@ -20,39 +20,39 @@
} }
</script> </script>
<form action="/cart" method="post" id="cartform"> <form action="/cart" method="post" id="cartform">
<table id="basket"> <table id="basket">
<tr> <tr>
<th>Item Description</th> <th>Item Description</th>
<th>Price</th> <th>Price</th>
<th>Qty</th> <th>Qty</th>
<th>Delete</th> <th>Delete</th>
<th>Total</th> <th>Total</th>
</tr>{% for item in cart.items %} </tr>{% for item in cart.items %}
<tr class="basket-{% cycle 'odd', 'even' %}"> <tr class="basket-{% cycle 'odd', 'even' %}">
<td class="basket-column-one"> <td class="basket-column-one">
<div class="basket-images"> <div class="basket-images">
<a href="{{ item.product.url }}" title="{{ item.title | escape }} &mdash; {{ item.product.description | strip_html | truncate: 50 | escape }}"><img src="{{ item.product.images.first | product_img_url: 'thumb' }}" alt="{{ item.title | escape }}" /></a> <a href="{{ item.product.url }}" title="{{ item.title | escape }} &mdash; {{ item.product.description | strip_html | truncate: 50 | escape }}"><img src="{{ item.product.images.first | product_img_url: 'thumb' }}" alt="{{ item.title | escape }}" /></a>
</div> </div>
<div class="basket-desc"> <div class="basket-desc">
<p><a href="{{ item.product.url }}">{{ item.title }}</a></p> <p><a href="{{ item.product.url }}">{{ item.title }}</a></p>
{{ item.product.description | strip_html | truncate: 120 }} {{ item.product.description | strip_html | truncate: 120 }}
</div> </div>
</td> </td>
<td class="basket-column">{{ item.price | money }}{% if item.variant.compare_at_price > item.price %}<br /><del>{{ item.variant.compare_at_price | money }}</del>{% endif %}</td> <td class="basket-column">{{ item.price | money }}{% if item.variant.compare_at_price > item.price %}<br /><del>{{ item.variant.compare_at_price | money }}</del>{% endif %}</td>
<td class="basket-column"><input type="text" size="4" name="updates[{{item.variant.id}}]" id="updates_{{ item.variant.id }}" value="{{ item.quantity }}" onfocus="this.select();"/></td> <td class="basket-column"><input type="text" size="4" name="updates[{{item.variant.id}}]" id="updates_{{ item.variant.id }}" value="{{ item.quantity }}" onfocus="this.select();"/></td>
<td class="basket-column"><a href="#" onclick="remove_item({{ item.variant.id }}); return false;">Remove</a></td> <td class="basket-column"><a href="#" onclick="remove_item({{ item.variant.id }}); return false;">Remove</a></td>
<td class="basket-column">{{ item.line_price | money }}</td> <td class="basket-column">{{ item.line_price | money }}</td>
</tr>{% endfor %} </tr>{% endfor %}
</table> </table>
<div id="basket-right"> <div id="basket-right">
<h3>Subtotal {{ cart.total_price | money }}</h3> <h3>Subtotal {{ cart.total_price | money }}</h3>
<input type="image" src="{{ 'update.png' | asset_url }}" id="update-cart" name="update" value="Update" /> <input type="image" src="{{ 'update.png' | asset_url }}" id="update-cart" name="update" value="Update" />
<input type="image" src="{{ 'checkout.png' | asset_url }}" name="checkout" value="Checkout" /> <input type="image" src="{{ 'checkout.png' | asset_url }}" name="checkout" value="Checkout" />
{% if additional_checkout_buttons %} {% if additional_checkout_buttons %}
<div class="additional-checkout-buttons"> <div class="additional-checkout-buttons">
<p>- or -</p> <p>- or -</p>
{{ content_for_additional_checkout_buttons }} {{ content_for_additional_checkout_buttons }}
</div> </div>
{% endif %} {% endif %}
</div> </div>
</form>{% endif %} </form>{% endif %}

View File

@@ -1,19 +1,19 @@
{% paginate collection.products by 12 %}{% if collection.products.size == 0 %} {% paginate collection.products by 12 %}{% if collection.products.size == 0 %}
<strong>No products found in this collection.</strong>{% else %} <strong>No products found in this collection.</strong>{% else %}
<h1>{{ collection.title }}</h1> <h1>{{ collection.title }}</h1>
{{ collection.description }} {{ collection.description }}
<table id="gallery"> <table id="gallery">
{% tablerow product in collection.products cols: 3 %} {% tablerow product in collection.products cols: 3 %}
<div class="gallery-image"> <div class="gallery-image">
<a href="{{ product.url | within: collection }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 | escape }}"><img src="{{ product.images.first | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ product.url | within: collection }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 | escape }}"><img src="{{ product.images.first | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a>
</div> </div>
<div class="gallery-info"> <div class="gallery-info">
<a href="{{ product.url | within: collection }}">{{ product.title | truncate: 30 }}</a><br /> <a href="{{ product.url | within: collection }}">{{ product.title | truncate: 30 }}</a><br />
<small>{{ product.price | money }}{% if product.compare_at_price_max > product.price %} <del>{{ product.compare_at_price_max | money }}</del>{% endif %}</small> <small>{{ product.price | money }}{% if product.compare_at_price_max > product.price %} <del>{{ product.compare_at_price_max | money }}</del>{% endif %}</small>
</div> </div>
{% endtablerow %} {% endtablerow %}
</table>{% if paginate.pages > 1 %} </table>{% if paginate.pages > 1 %}
<div id="paginate"> <div id="paginate">
{{ paginate | default_pagination }} {{ paginate | default_pagination }}
</div>{% endif %}{% endif %} </div>{% endif %}{% endif %}
{% endpaginate %} {% endpaginate %}

View File

@@ -1,22 +1,22 @@
<div id="about-excerpt"> <div id="about-excerpt">
{% assign article = pages.frontpage %} {% assign article = pages.frontpage %}
{% if article.content != "" %} {% if article.content != "" %}
<h2>{{ article.title }}</h2> <h2>{{ article.title }}</h2>
{{ article.content }} {{ article.content }}
{% else %} {% else %}
In <em>Admin &gt; Blogs &amp; Pages</em>, create a page with the handle <strong><code>frontpage</code></strong> and it will show up here.<br /> In <em>Admin &gt; Blogs &amp; Pages</em>, create a page with the handle <strong><code>frontpage</code></strong> and it will show up here.<br />
{{ "Learn more about handles" | link_to "http://wiki.shopify.com/Handle" }} {{ "Learn more about handles" | link_to "http://wiki.shopify.com/Handle" }}
{% endif %} {% endif %}
</div> </div>
<table id="gallery"> <table id="gallery">
{% tablerow product in collections.frontpage.products cols: 3 limit: 12 %} {% tablerow product in collections.frontpage.products cols: 3 limit: 12 %}
<div class="gallery-image"> <div class="gallery-image">
<a href="{{ product.url | within: collections.frontpage }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 | escape }}"><img src="{{ product.images.first | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ product.url | within: collections.frontpage }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 | escape }}"><img src="{{ product.images.first | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a>
</div> </div>
<div class="gallery-info"> <div class="gallery-info">
<a href="{{ product.url | within: collections.frontpage }}">{{ product.title | truncate: 30 }}</a><br /> <a href="{{ product.url | within: collections.frontpage }}">{{ product.title | truncate: 30 }}</a><br />
<small>{{ product.price | money }}{% if product.compare_at_price_max > product.price %} <del>{{ product.compare_at_price_max | money }}</del>{% endif %}</small> <small>{{ product.price | money }}{% if product.compare_at_price_max > product.price %} <del>{{ product.compare_at_price_max | money }}</del>{% endif %}</small>
</div> </div>
{% endtablerow %} {% endtablerow %}
</table> </table>

View File

@@ -1,3 +1,3 @@
<h1>{{ page.title }}</h1> <h1>{{ page.title }}</h1>
{{ page.content }} {{ page.content }}

View File

@@ -1,19 +1,19 @@
<div id="product-left"> <div id="product-left">
{% for image in product.images %}{% if forloop.first %}<div id="product-image"> {% for image in product.images %}{% if forloop.first %}<div id="product-image">
<a href="{{ image | product_img_url: 'large' }}" rel="lightbox[images]" title="{{ product.title | escape }}"><img src="{{ image | product_img_url: 'medium' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ image | product_img_url: 'large' }}" rel="lightbox[images]" title="{{ product.title | escape }}"><img src="{{ image | product_img_url: 'medium' }}" alt="{{ product.title | escape }}" /></a>
</div>{% else %} </div>{% else %}
<div class="product-images"> <div class="product-images">
<a href="{{ image | product_img_url: 'large' }}" rel="lightbox[images]" title="{{ product.title | escape }}"><img src="{{ image | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ image | product_img_url: 'large' }}" rel="lightbox[images]" title="{{ product.title | escape }}"><img src="{{ image | product_img_url: 'small' }}" alt="{{ product.title | escape }}" /></a>
</div>{% endif %}{% endfor %} </div>{% endif %}{% endfor %}
</div> </div>
<div id="product-right"> <div id="product-right">
<h1>{{ product.title }}</h1> <h1>{{ product.title }}</h1>
{{ product.description }} {{ product.description }}
{% if product.available %}
<form action="/cart/add" method="post">
<div id="product-variants"> {% if product.available %}
<form action="/cart/add" method="post">
<div id="product-variants">
<div id="price-field"></div> <div id="price-field"></div>
<select id="product-select" name='id'> <select id="product-select" name='id'>
@@ -21,18 +21,18 @@
<option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option> <option value="{{ variant.id }}">{{ variant.title }} - {{ variant.price | money }}</option>
{% endfor %} {% endfor %}
</select> </select>
</div> </div>
<input type="image" src="{{ 'purchase.png' | asset_url }}" name="add" value="Purchase" id="purchase" /> <input type="image" src="{{ 'purchase.png' | asset_url }}" name="add" value="Purchase" id="purchase" />
</form> </form>
{% else %} {% else %}
<p class="bold-red">This product is temporarily unavailable</p> <p class="bold-red">This product is temporarily unavailable</p>
{% endif %} {% endif %}
<div id="product-details"> <div id="product-details">
<strong>Continue Shopping</strong><br /> <strong>Continue Shopping</strong><br />
Browse more {{ product.type | link_to_type }} or additional {{ product.vendor | link_to_vendor }} products. Browse more {{ product.type | link_to_type }} or additional {{ product.vendor | link_to_vendor }} products.
</div> </div>
</div> </div>
@@ -48,14 +48,14 @@
} else { } else {
// variant doesn't exist // variant doesn't exist
$('purchase').addClass('disabled'); // set add-to-cart button to unavailable class $('purchase').addClass('disabled'); // set add-to-cart button to unavailable class
$('purchase').disabled = true; // disable add-to-cart button $('purchase').disabled = true; // disable add-to-cart button
$('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message $('price-field').innerHTML = (variant) ? "Sold Out" : "Unavailable"; // update price-field message
} }
}; };
// initialize multi selector for product // initialize multi selector for product
window.addEvent('domready', function() { window.addEvent('domready', function() {
new Shopify.OptionSelectors("product-select", { product: {{ product | json }}, onVariantSelected: selectCallback }); new Shopify.OptionSelectors("product-select", { product: {{ product | json }}, onVariantSelected: selectCallback });
}); });
--> -->
</script> </script>

View File

@@ -22,101 +22,101 @@
<body id="page-{{ template }}"> <body id="page-{{ template }}">
<div id="header"> <div id="header">
<div class="container"> <div class="container">
<div id="logo"> <div id="logo">
<h1><a href="/" title="{{ shop.name }}">{{ shop.name }}</a></h1> <h1><a href="/" title="{{ shop.name }}">{{ shop.name }}</a></h1>
</div> </div>
<div id="navigation"> <div id="navigation">
<ul id="navigate"> <ul id="navigate">
<li><a href="/cart">View Cart</a></li> <li><a href="/cart">View Cart</a></li>
{% for link in linklists.main-menu.links reversed %} {% for link in linklists.main-menu.links reversed %}
<li><a href="{{ link.url }}">{{ link.title }}</a></li> <li><a href="{{ link.url }}">{{ link.title }}</a></li>
{% endfor %} {% endfor %}
</ul> </ul>
</div> </div>
</div> </div>
</div> </div>
<div id="mini-header"> <div id="mini-header">
<div class="container"> <div class="container">
<div id="shopping-cart"> <div id="shopping-cart">
<a href="/cart">Your shopping cart contains {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }}</a> <a href="/cart">Your shopping cart contains {{ cart.item_count }} {{ cart.item_count | pluralize: 'item', 'items' }}</a>
</div> </div>
<div id="search-box"> <div id="search-box">
<form action="/search" method="get"> <form action="/search" method="get">
<input type="text" name="q" id="q" /> <input type="text" name="q" id="q" />
<input type="image" src="{{ 'seek.png' | asset_url }}" value="Seek" onclick="this.parentNode.submit(); return false;" id="seek" /> <input type="image" src="{{ 'seek.png' | asset_url }}" value="Seek" onclick="this.parentNode.submit(); return false;" id="seek" />
</form> </form>
</div> </div>
</div> </div>
</div> </div>
<div id="layout"> <div id="layout">
<div class="container"> <div class="container">
<div id="layout-left" {% if template != "cart" %}{% if template != "product" %}style="width:619px"{% endif %}{% endif %}>{% if template == "search" %} <div id="layout-left" {% if template != "cart" %}{% if template != "product" %}style="width:619px"{% endif %}{% endif %}>{% if template == "search" %}
<h1>Search Results</h1>{% endif %} <h1>Search Results</h1>{% endif %}
{{ content_for_layout }} {{ content_for_layout }}
</div>{% if template != "cart" %}{% if template != "product" %} </div>{% if template != "cart" %}{% if template != "product" %}
<div id="layout-right"> <div id="layout-right">
{% if template == "index" %} {% if template == "index" %}
{% if blogs.news.articles.size > 1 %} {% if blogs.news.articles.size > 1 %}
<a href="{{ shop.url }}/blogs/news.xml"><img src="{{ 'feed.png' | asset_url }}" alt="Subscribe" class="feed" /></a> <a href="{{ shop.url }}/blogs/news.xml"><img src="{{ 'feed.png' | asset_url }}" alt="Subscribe" class="feed" /></a>
<h3><a href="/blogs/news">More news</a></h3> <h3><a href="/blogs/news">More news</a></h3>
<ul id="blogs">{% for article in blogs.news.articles limit: 6 offset: 1 %} <ul id="blogs">{% for article in blogs.news.articles limit: 6 offset: 1 %}
<li><a href="{{ article.url }}">{{ article.title | strip_html | truncate: 30 }}</a><br /> <li><a href="{{ article.url }}">{{ article.title | strip_html | truncate: 30 }}</a><br />
<small>{{ article.content | strip_html | truncatewords: 12 }}</small> <small>{{ article.content | strip_html | truncatewords: 12 }}</small>
</li>{% endfor %} </li>{% endfor %}
</ul> </ul>
{% endif %} {% endif %}
{% endif %} {% endif %}
{% if template == "collection" %} {% if template == "collection" %}
<h3>Collection Tags</h3> <h3>Collection Tags</h3>
<div id="tags">{% if collection.tags.size == 0 %} <div id="tags">{% if collection.tags.size == 0 %}
No tags found.{% else %} No tags found.{% else %}
<span class="tags">{% for tag in collection.tags %}{% if current_tags contains tag %} {{ tag | highlight_active_tag | link_to_remove_tag: tag }}{% else %} {{ tag | highlight_active_tag | link_to_add_tag: tag }}{% endif %}{% unless forloop.last %}, {% endunless %}{% endfor %}</span>{% endif %} <span class="tags">{% for tag in collection.tags %}{% if current_tags contains tag %} {{ tag | highlight_active_tag | link_to_remove_tag: tag }}{% else %} {{ tag | highlight_active_tag | link_to_add_tag: tag }}{% endif %}{% unless forloop.last %}, {% endunless %}{% endfor %}</span>{% endif %}
</div> </div>
{% endif %} {% endif %}
<h3>Navigation</h3> <h3>Navigation</h3>
<ul id="links"> <ul id="links">
{% for link in linklists.main-menu.links %} {% for link in linklists.main-menu.links %}
<li><a href="{{ link.url }}">{{ link.title }}</a></li> <li><a href="{{ link.url }}">{{ link.title }}</a></li>
{% endfor %} {% endfor %}
</ul> </ul>
{% if template != "page" %} {% if template != "page" %}
<h3>Featured Products</h3> <h3>Featured Products</h3>
<ul id="featuring">{% for product in collections.frontpage.products limit: 6 %} <ul id="featuring">{% for product in collections.frontpage.products limit: 6 %}
<li class="featuring-list"> <li class="featuring-list">
<div class="featuring-image"> <div class="featuring-image">
<a href="{{ product.url | within: collections.frontpage }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 }}"><img src="{{ product.images.first | product_img_url: 'icon' }}" alt="{{ product.title | escape }}" /></a> <a href="{{ product.url | within: collections.frontpage }}" title="{{ product.title | escape }} &mdash; {{ product.description | strip_html | truncate: 50 }}"><img src="{{ product.images.first | product_img_url: 'icon' }}" alt="{{ product.title | escape }}" /></a>
</div> </div>
<div class="featuring-info"> <div class="featuring-info">
<a href="{{ product.url | within: collections.frontpage }}">{{ product.title | strip_html | truncate: 28 }}</a><br /> <a href="{{ product.url | within: collections.frontpage }}">{{ product.title | strip_html | truncate: 28 }}</a><br />
<small><span class="light">from</span> {{ product.price | money }}</small> <small><span class="light">from</span> {{ product.price | money }}</small>
</div> </div>
</li>{% endfor %} </li>{% endfor %}
</ul> </ul>
{% endif %} {% endif %}
</div>{% endif %}{% endif %} </div>{% endif %}{% endif %}
</div> </div>
</div> </div>
<div id="footer"> <div id="footer">
<div id="footer-fader"> <div id="footer-fader">
<div class="container"> <div class="container">
<div id="footer-right">{% for link in linklists.footer.links %} <div id="footer-right">{% for link in linklists.footer.links %}
{{ link.title | link_to: link.url }} {% unless forloop.last %}&#124;{% endunless %}{% endfor %} {{ link.title | link_to: link.url }} {% unless forloop.last %}&#124;{% endunless %}{% endfor %}
</div> </div>
<span id="footer-left"> <span id="footer-left">
Copyright &copy; {{ "now" | date: "%Y" }} <a href="/">{{ shop.name }}</a>. All Rights Reserved. All prices {{ shop.currency }}.<br /> Copyright &copy; {{ "now" | date: "%Y" }} <a href="/">{{ shop.name }}</a>. All Rights Reserved. All prices {{ shop.currency }}.<br />
This website is powered by <a href="http://www.shopify.com">Shopify</a>. This website is powered by <a href="http://www.shopify.com">Shopify</a>.
</span> </span>
</div> </div>
</div> </div>
</div> </div>
</body> </body>
</html> </html>

View File

@@ -1,10 +1,10 @@
# This profiler run simulates Shopify. # This profiler run simulates Shopify.
# We are looking in the tests directory for liquid files and render them within the designated layout file. # We are looking in the tests directory for liquid files and render them within the designated layout file.
# We will also export a substantial database to liquid which the templates can render values of. # We will also export a substantial database to liquid which the templates can render values of.
# All this is to make the benchmark as non syntetic as possible. All templates and tests are lifted from # All this is to make the benchmark as non syntetic as possible. All templates and tests are lifted from
# direct real-world usage and the profiler measures code that looks very similar to the way it looks in # direct real-world usage and the profiler measures code that looks very similar to the way it looks in
# Shopify which is likely the biggest user of liquid in the world which something to the tune of several # Shopify which is likely the biggest user of liquid in the world which something to the tune of several
# million Template#render calls a day. # million Template#render calls a day.
require 'rubygems' require 'rubygems'
require 'active_support' require 'active_support'
@@ -14,9 +14,20 @@ require File.dirname(__FILE__) + '/shopify/liquid'
require File.dirname(__FILE__) + '/shopify/database.rb' require File.dirname(__FILE__) + '/shopify/database.rb'
class ThemeRunner class ThemeRunner
class FileSystem
# Load all templates into memory, do this now so that def initialize(path)
# we don't profile IO. @path = path
end
# Called by Liquid to retrieve a template file
def read_template_file(template_path, context)
File.read(@path + '/' + template_path + '.liquid')
end
end
# Load all templates into memory, do this now so that
# we don't profile IO.
def initialize def initialize
@tests = Dir[File.dirname(__FILE__) + '/tests/**/*.liquid'].collect do |test| @tests = Dir[File.dirname(__FILE__) + '/tests/**/*.liquid'].collect do |test|
next if File.basename(test) == 'theme.liquid' next if File.basename(test) == 'theme.liquid'
@@ -25,7 +36,7 @@ class ThemeRunner
[File.read(test), (File.file?(theme_path) ? File.read(theme_path) : nil), test] [File.read(test), (File.file?(theme_path) ? File.read(theme_path) : nil), test]
end.compact end.compact
end end
def compile def compile
# Dup assigns because will make some changes to them # Dup assigns because will make some changes to them
@@ -47,8 +58,8 @@ class ThemeRunner
# Compute page_tempalte outside of profiler run, uninteresting to profiler # Compute page_tempalte outside of profiler run, uninteresting to profiler
page_template = File.basename(template_name, File.extname(template_name)) page_template = File.basename(template_name, File.extname(template_name))
compile_and_render(liquid, layout, assigns, page_template) compile_and_render(liquid, layout, assigns, page_template, template_name)
end end
end end
@@ -74,13 +85,13 @@ class ThemeRunner
html = nil html = nil
RubyProf.resume RubyProf.resume
html = compile_and_render(liquid, layout, assigns, page_template) html = compile_and_render(liquid, layout, assigns, page_template, template_name)
RubyProf.pause RubyProf.pause
# return the result and the MD5 of the content, this can be used to detect regressions between liquid version # return the result and the MD5 of the content, this can be used to detect regressions between liquid version
$stdout.puts "* rendered template %s, content: %s" % [template_name, Digest::MD5.hexdigest(html)] $stdout.puts "* rendered template %s, content: %s" % [template_name, Digest::MD5.hexdigest(html)]
# Uncomment to dump html files to /tmp so that you can inspect for errors # Uncomment to dump html files to /tmp so that you can inspect for errors
# File.open("/tmp/#{File.basename(template_name)}.html", "w+") { |fp| fp <<html} # File.open("/tmp/#{File.basename(template_name)}.html", "w+") { |fp| fp <<html}
end end
@@ -88,19 +99,20 @@ class ThemeRunner
RubyProf.stop RubyProf.stop
end end
def compile_and_render(template, layout, assigns, page_template) def compile_and_render(template, layout, assigns, page_template, template_file)
tmpl = Liquid::Template.new tmpl = Liquid::Template.new
tmpl.assigns['page_title'] = 'Page title' tmpl.assigns['page_title'] = 'Page title'
tmpl.assigns['template'] = page_template tmpl.assigns['template'] = page_template
tmpl.registers[:file_system] = ThemeRunner::FileSystem.new(File.dirname(template_file))
content_for_layout = tmpl.parse(template).render(assigns) content_for_layout = tmpl.parse(template).render(assigns)
if layout if layout
assigns['content_for_layout'] = content_for_layout assigns['content_for_layout'] = content_for_layout
tmpl.parse(layout).render(assigns) tmpl.parse(layout).render(assigns)
else else
content_for_layout content_for_layout
end end
end end
end end

View File

@@ -12,7 +12,7 @@ class AssignTest < Test::Unit::TestCase
'{% assign foo = values %}.{{ foo[1] }}.', '{% assign foo = values %}.{{ foo[1] }}.',
'values' => %w{foo bar baz}) 'values' => %w{foo bar baz})
end end
def test_assign_with_filter def test_assign_with_filter
assert_template_result('.bar.', assert_template_result('.bar.',
'{% assign foo = values | split: "," %}.{{ foo[1] }}.', '{% assign foo = values | split: "," %}.{{ foo[1] }}.',

View File

@@ -124,4 +124,4 @@ class ConditionTest < Test::Unit::TestCase
assert !Condition.new(left, op, right).evaluate(@context || Liquid::Context.new), assert !Condition.new(left, op, right).evaluate(@context || Liquid::Context.new),
"Evaluated true: #{left} #{op} #{right}" "Evaluated true: #{left} #{op} #{right}"
end end
end # ConditionTest end # ConditionTest

View File

@@ -0,0 +1,25 @@
require 'test_helper'
module MoneyFilter
def money(input)
sprintf(' %d$ ', input)
end
end
module CanadianMoneyFilter
def money(input)
sprintf(' %d$ CAD ', input)
end
end
class HashOrderingTest < Test::Unit::TestCase
include Liquid
def test_global_register_order
Template.register_filter(MoneyFilter)
Template.register_filter(CanadianMoneyFilter)
assert_equal " 1000$ CAD ", Template.parse("{{1000 | money}}").render(nil, nil)
end
end

View File

@@ -49,7 +49,7 @@ class OutputTest < Test::Unit::TestCase
expected = %| good bad good | expected = %| good bad good |
assert_equal expected, Template.parse(text).render(@assigns) assert_equal expected, Template.parse(text).render(@assigns)
end end
def test_variable_piping def test_variable_piping
text = %( {{ car.gm | make_funny }} ) text = %( {{ car.gm | make_funny }} )
@@ -113,4 +113,4 @@ class OutputTest < Test::Unit::TestCase
assert_equal expected, Template.parse(text).render(@assigns, :filters => [FunnyFilter]) assert_equal expected, Template.parse(text).render(@assigns, :filters => [FunnyFilter])
end end
end # OutputTest end # OutputTest

View File

@@ -1,3 +1,5 @@
# encoding: utf-8
require 'test_helper' require 'test_helper'
class Filters class Filters
@@ -58,6 +60,7 @@ class StandardFiltersTest < Test::Unit::TestCase
assert_equal '1234567890', @filters.truncate('1234567890', 20) assert_equal '1234567890', @filters.truncate('1234567890', 20)
assert_equal '...', @filters.truncate('1234567890', 0) assert_equal '...', @filters.truncate('1234567890', 0)
assert_equal '1234567890', @filters.truncate('1234567890') assert_equal '1234567890', @filters.truncate('1234567890')
assert_equal "测试...", @filters.truncate("测试测试测试测试", 5)
end end
def test_strip def test_strip
@@ -82,12 +85,16 @@ class StandardFiltersTest < Test::Unit::TestCase
assert_equal 'one two...', @filters.truncatewords('one two three', 2) assert_equal 'one two...', @filters.truncatewords('one two three', 2)
assert_equal 'one two three', @filters.truncatewords('one two three') assert_equal 'one two three', @filters.truncatewords('one two three')
assert_equal 'Two small (13&#8221; x 5.5&#8221; x 10&#8221; high) baskets fit inside one large basket (13&#8221;...', @filters.truncatewords('Two small (13&#8221; x 5.5&#8221; x 10&#8221; high) baskets fit inside one large basket (13&#8221; x 16&#8221; x 10.5&#8221; high) with cover.', 15) assert_equal 'Two small (13&#8221; x 5.5&#8221; x 10&#8221; high) baskets fit inside one large basket (13&#8221;...', @filters.truncatewords('Two small (13&#8221; x 5.5&#8221; x 10&#8221; high) baskets fit inside one large basket (13&#8221; x 16&#8221; x 10.5&#8221; high) with cover.', 15)
assert_equal "测试测试测试测试", @filters.truncatewords('测试测试测试测试', 5)
end end
def test_strip_html def test_strip_html
assert_equal 'test', @filters.strip_html("<div>test</div>") assert_equal 'test', @filters.strip_html("<div>test</div>")
assert_equal 'test', @filters.strip_html("<div id='test'>test</div>") assert_equal 'test', @filters.strip_html("<div id='test'>test</div>")
assert_equal '', @filters.strip_html("<script type='text/javascript'>document.write('some stuff');</script>") assert_equal '', @filters.strip_html("<script type='text/javascript'>document.write('some stuff');</script>")
assert_equal '', @filters.strip_html("<style type='text/css'>foo bar</style>")
assert_equal 'test', @filters.strip_html("<div\nclass='multiline'>test</div>")
assert_equal 'test', @filters.strip_html("<!-- foo bar \n test -->test")
assert_equal '', @filters.strip_html(nil) assert_equal '', @filters.strip_html(nil)
end end
@@ -101,6 +108,10 @@ class StandardFiltersTest < Test::Unit::TestCase
assert_equal [{"a" => 1}, {"a" => 2}, {"a" => 3}, {"a" => 4}], @filters.sort([{"a" => 4}, {"a" => 3}, {"a" => 1}, {"a" => 2}], "a") assert_equal [{"a" => 1}, {"a" => 2}, {"a" => 3}, {"a" => 4}], @filters.sort([{"a" => 4}, {"a" => 3}, {"a" => 1}, {"a" => 2}], "a")
end end
def test_reverse
assert_equal [4,3,2,1], @filters.reverse([1,2,3,4])
end
def test_map def test_map
assert_equal [1,2,3,4], @filters.map([{"a" => 1}, {"a" => 2}, {"a" => 3}, {"a" => 4}], 'a') assert_equal [1,2,3,4], @filters.map([{"a" => 1}, {"a" => 2}, {"a" => 3}, {"a" => 4}], 'a')
assert_template_result 'abc', "{{ ary | map:'foo' | map:'bar' }}", assert_template_result 'abc', "{{ ary | map:'foo' | map:'bar' }}",
@@ -141,11 +152,15 @@ class StandardFiltersTest < Test::Unit::TestCase
assert_equal '07/05/2006', @filters.date("2006-07-05 10:00:00", "%m/%d/%Y") assert_equal '07/05/2006', @filters.date("2006-07-05 10:00:00", "%m/%d/%Y")
assert_equal "07/16/2004", @filters.date("Fri Jul 16 01:00:00 2004", "%m/%d/%Y") assert_equal "07/16/2004", @filters.date("Fri Jul 16 01:00:00 2004", "%m/%d/%Y")
assert_equal "#{Date.today.year}", @filters.date('now', '%Y')
assert_equal "#{Date.today.year}", @filters.date('today', '%Y')
assert_equal nil, @filters.date(nil, "%B") assert_equal nil, @filters.date(nil, "%B")
assert_equal "07/05/2006", @filters.date(1152098955, "%m/%d/%Y") with_timezone("UTC") do
assert_equal "07/05/2006", @filters.date("1152098955", "%m/%d/%Y") assert_equal "07/05/2006", @filters.date(1152098955, "%m/%d/%Y")
assert_equal "07/05/2006", @filters.date("1152098955", "%m/%d/%Y")
end
end end
@@ -174,6 +189,7 @@ class StandardFiltersTest < Test::Unit::TestCase
def test_strip_newlines def test_strip_newlines
assert_template_result 'abc', "{{ source | strip_newlines }}", 'source' => "a\nb\nc" assert_template_result 'abc', "{{ source | strip_newlines }}", 'source' => "a\nb\nc"
assert_template_result 'abc', "{{ source | strip_newlines }}", 'source' => "a\r\nb\nc"
end end
def test_newlines_to_br def test_newlines_to_br
@@ -198,6 +214,8 @@ class StandardFiltersTest < Test::Unit::TestCase
assert_match(/(6\.3)|(6\.(0{13})1)/, Template.parse("{{ '2.1' | times:3 }}").render) assert_match(/(6\.3)|(6\.(0{13})1)/, Template.parse("{{ '2.1' | times:3 }}").render)
assert_template_result "6", "{{ '2.1' | times:3 | replace: '.','-' | plus:0}}" assert_template_result "6", "{{ '2.1' | times:3 | replace: '.','-' | plus:0}}"
assert_template_result "7.25", "{{ 0.0725 | times:100 }}"
end end
def test_divided_by def test_divided_by
@@ -209,6 +227,8 @@ class StandardFiltersTest < Test::Unit::TestCase
assert_template_result "5", "{{ 15 | divided_by:3 }}" assert_template_result "5", "{{ 15 | divided_by:3 }}"
assert_template_result "Liquid error: divided by 0", "{{ 5 | divided_by:0 }}" assert_template_result "Liquid error: divided by 0", "{{ 5 | divided_by:0 }}"
assert_template_result "0.5", "{{ 2.0 | divided_by:4 }}"
end end
def test_modulo def test_modulo
@@ -230,4 +250,14 @@ class StandardFiltersTest < Test::Unit::TestCase
def test_cannot_access_private_methods def test_cannot_access_private_methods
assert_template_result('a',"{{ 'a' | to_number }}") assert_template_result('a',"{{ 'a' | to_number }}")
end end
private
def with_timezone(tz)
old_tz = ENV['TZ']
ENV['TZ'] = tz
yield
ensure
ENV['TZ'] = old_tz
end
end # StandardFiltersTest end # StandardFiltersTest

View File

@@ -187,16 +187,16 @@ HERE
expected = "1234" expected = "1234"
assert_template_result(expected,markup,assigns) assert_template_result(expected,markup,assigns)
# tests to ensure it only breaks out of the local for loop # tests to ensure it only breaks out of the local for loop
# and not all of them. # and not all of them.
assigns = {'array' => [[1,2],[3,4],[5,6]] } assigns = {'array' => [[1,2],[3,4],[5,6]] }
markup = '{% for item in array %}' + markup = '{% for item in array %}' +
'{% for i in item %}' + '{% for i in item %}' +
'{% if i == 1 %}' + '{% if i == 1 %}' +
'{% break %}' + '{% break %}' +
'{% endif %}' + '{% endif %}' +
'{{ i }}' + '{{ i }}' +
'{% endfor %}' + '{% endfor %}' +
'{% endfor %}' '{% endfor %}'
expected = '3456' expected = '3456'
assert_template_result(expected, markup, assigns) assert_template_result(expected, markup, assigns)
@@ -213,15 +213,15 @@ HERE
markup = '{% for i in array.items %}{% continue %}{% endfor %}' markup = '{% for i in array.items %}{% continue %}{% endfor %}'
expected = "" expected = ""
assert_template_result(expected,markup,assigns) assert_template_result(expected,markup,assigns)
markup = '{% for i in array.items %}{{ i }}{% continue %}{% endfor %}' markup = '{% for i in array.items %}{{ i }}{% continue %}{% endfor %}'
expected = "12345" expected = "12345"
assert_template_result(expected,markup,assigns) assert_template_result(expected,markup,assigns)
markup = '{% for i in array.items %}{% continue %}{{ i }}{% endfor %}' markup = '{% for i in array.items %}{% continue %}{{ i }}{% endfor %}'
expected = "" expected = ""
assert_template_result(expected,markup,assigns) assert_template_result(expected,markup,assigns)
markup = '{% for i in array.items %}{% if i > 3 %}{% continue %}{% endif %}{{ i }}{% endfor %}' markup = '{% for i in array.items %}{% if i > 3 %}{% continue %}{% endif %}{{ i }}{% endfor %}'
expected = "123" expected = "123"
@@ -233,13 +233,13 @@ HERE
# tests to ensure it only continues the local for loop and not all of them. # tests to ensure it only continues the local for loop and not all of them.
assigns = {'array' => [[1,2],[3,4],[5,6]] } assigns = {'array' => [[1,2],[3,4],[5,6]] }
markup = '{% for item in array %}' + markup = '{% for item in array %}' +
'{% for i in item %}' + '{% for i in item %}' +
'{% if i == 1 %}' + '{% if i == 1 %}' +
'{% continue %}' + '{% continue %}' +
'{% endif %}' + '{% endif %}' +
'{{ i }}' + '{{ i }}' +
'{% endfor %}' + '{% endfor %}' +
'{% endfor %}' '{% endfor %}'
expected = '23456' expected = '23456'
assert_template_result(expected, markup, assigns) assert_template_result(expected, markup, assigns)
@@ -253,32 +253,45 @@ HERE
def test_for_tag_string def test_for_tag_string
# ruby 1.8.7 "String".each => Enumerator with single "String" element. # ruby 1.8.7 "String".each => Enumerator with single "String" element.
# ruby 1.9.3 no longer supports .each on String though we mimic # ruby 1.9.3 no longer supports .each on String though we mimic
# the functionality for backwards compatibility # the functionality for backwards compatibility
assert_template_result('test string', assert_template_result('test string',
'{%for val in string%}{{val}}{%endfor%}', '{%for val in string%}{{val}}{%endfor%}',
'string' => "test string") 'string' => "test string")
assert_template_result('test string', assert_template_result('test string',
'{%for val in string limit:1%}{{val}}{%endfor%}', '{%for val in string limit:1%}{{val}}{%endfor%}',
'string' => "test string") 'string' => "test string")
assert_template_result('val-string-1-1-0-1-0-true-true-test string', assert_template_result('val-string-1-1-0-1-0-true-true-test string',
'{%for val in string%}' + '{%for val in string%}' +
'{{forloop.name}}-' + '{{forloop.name}}-' +
'{{forloop.index}}-' + '{{forloop.index}}-' +
'{{forloop.length}}-' + '{{forloop.length}}-' +
'{{forloop.index0}}-' + '{{forloop.index0}}-' +
'{{forloop.rindex}}-' + '{{forloop.rindex}}-' +
'{{forloop.rindex0}}-' + '{{forloop.rindex0}}-' +
'{{forloop.first}}-' + '{{forloop.first}}-' +
'{{forloop.last}}-' + '{{forloop.last}}-' +
'{{val}}{%endfor%}', '{{val}}{%endfor%}',
'string' => "test string") 'string' => "test string")
end end
def test_blank_string_not_iterable def test_blank_string_not_iterable
assert_template_result('', "{% for char in characters %}I WILL NOT BE OUTPUT{% endfor %}", 'characters' => '') assert_template_result('', "{% for char in characters %}I WILL NOT BE OUTPUT{% endfor %}", 'characters' => '')
end end
def test_bad_variable_naming_in_for_loop
assert_raise(Liquid::SyntaxError) do
Liquid::Template.parse('{% for a/b in x %}{% endfor %}')
end
end
def test_spacing_with_variable_naming_in_for_loop
expected = '12345'
template = '{% for item in items %}{{item}}{% endfor %}'
assigns = {'items' => [1,2,3,4,5]}
assert_template_result(expected, template, assigns)
end
end end

View File

@@ -42,11 +42,11 @@ class HtmlTagTest < Test::Unit::TestCase
def test_quoted_fragment def test_quoted_fragment
assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td><td class=\"col2\"> 2 </td><td class=\"col3\"> 3 </td></tr>\n<tr class=\"row2\"><td class=\"col1\"> 4 </td><td class=\"col2\"> 5 </td><td class=\"col3\"> 6 </td></tr>\n", assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td><td class=\"col2\"> 2 </td><td class=\"col3\"> 3 </td></tr>\n<tr class=\"row2\"><td class=\"col1\"> 4 </td><td class=\"col2\"> 5 </td><td class=\"col3\"> 6 </td></tr>\n",
"{% tablerow n in collections.frontpage cols:3%} {{n}} {% endtablerow %}", "{% tablerow n in collections.frontpage cols:3%} {{n}} {% endtablerow %}",
'collections' => {'frontpage' => [1,2,3,4,5,6]}) 'collections' => {'frontpage' => [1,2,3,4,5,6]})
assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td><td class=\"col2\"> 2 </td><td class=\"col3\"> 3 </td></tr>\n<tr class=\"row2\"><td class=\"col1\"> 4 </td><td class=\"col2\"> 5 </td><td class=\"col3\"> 6 </td></tr>\n", assert_template_result("<tr class=\"row1\">\n<td class=\"col1\"> 1 </td><td class=\"col2\"> 2 </td><td class=\"col3\"> 3 </td></tr>\n<tr class=\"row2\"><td class=\"col1\"> 4 </td><td class=\"col2\"> 5 </td><td class=\"col3\"> 6 </td></tr>\n",
"{% tablerow n in collections['frontpage'] cols:3%} {{n}} {% endtablerow %}", "{% tablerow n in collections['frontpage'] cols:3%} {{n}} {% endtablerow %}",
'collections' => {'frontpage' => [1,2,3,4,5,6]}) 'collections' => {'frontpage' => [1,2,3,4,5,6]})
end end
def test_enumerable_drop def test_enumerable_drop

View File

@@ -39,6 +39,15 @@ class OtherFileSystem
end end
end end
class CountingFileSystem
attr_reader :count
def read_template_file(template_path, context)
@count ||= 0
@count += 1
'from CountingFileSystem'
end
end
class IncludeTagTest < Test::Unit::TestCase class IncludeTagTest < Test::Unit::TestCase
include Liquid include Liquid
@@ -136,4 +145,22 @@ class IncludeTagTest < Test::Unit::TestCase
assert_equal "Product: Draft 151cm ", Template.parse("{% include template for product %}").render("template" => 'product', 'product' => { 'title' => 'Draft 151cm'}) assert_equal "Product: Draft 151cm ", Template.parse("{% include template for product %}").render("template" => 'product', 'product' => { 'title' => 'Draft 151cm'})
end end
end # IncludeTagTest
def test_include_tag_caches_second_read_of_same_partial
file_system = CountingFileSystem.new
assert_equal 'from CountingFileSystemfrom CountingFileSystem',
Template.parse("{% include 'pick_a_source' %}{% include 'pick_a_source' %}").render({}, :registers => {:file_system => file_system})
assert_equal 1, file_system.count
end
def test_include_tag_doesnt_cache_partials_across_renders
file_system = CountingFileSystem.new
assert_equal 'from CountingFileSystem',
Template.parse("{% include 'pick_a_source' %}").render({}, :registers => {:file_system => file_system})
assert_equal 1, file_system.count
assert_equal 'from CountingFileSystem',
Template.parse("{% include 'pick_a_source' %}").render({}, :registers => {:file_system => file_system})
assert_equal 2, file_system.count
end
end # IncludeTagTest

View File

@@ -9,7 +9,16 @@ class RawTagTest < Test::Unit::TestCase
end end
def test_output_in_raw def test_output_in_raw
assert_template_result '{{ test }}', assert_template_result '{{ test }}', '{% raw %}{{ test }}{% endraw %}'
'{% raw %}{{ test }}{% endraw %}' end
def test_open_tag_in_raw
assert_template_result ' Foobar {% invalid ', '{% raw %} Foobar {% invalid {% endraw %}'
assert_template_result ' Foobar invalid %} ', '{% raw %} Foobar invalid %} {% endraw %}'
assert_template_result ' Foobar {{ invalid ', '{% raw %} Foobar {{ invalid {% endraw %}'
assert_template_result ' Foobar invalid }} ', '{% raw %} Foobar invalid }} {% endraw %}'
assert_template_result ' Foobar {% invalid {% {% endraw ', '{% raw %} Foobar {% invalid {% {% endraw {% endraw %}'
assert_template_result ' Foobar {% {% {% ', '{% raw %} Foobar {% {% {% {% endraw %}'
assert_template_result ' test {% raw %} {% endraw %}', '{% raw %} test {% raw %} {% {% endraw %}endraw %}'
end end
end end

View File

@@ -1,5 +1,19 @@
require 'test_helper' require 'test_helper'
class TemplateContextDrop < Liquid::Drop
def before_method(method)
method
end
def foo
'fizzbuzz'
end
def baz
@context.registers['lulz']
end
end
class TemplateTest < Test::Unit::TestCase class TemplateTest < Test::Unit::TestCase
include Liquid include Liquid
@@ -71,4 +85,62 @@ class TemplateTest < Test::Unit::TestCase
assert_equal '1', t.render(assigns) assert_equal '1', t.render(assigns)
@global = nil @global = nil
end end
def test_resource_limits_render_length
t = Template.parse("0123456789")
t.resource_limits = { :render_length_limit => 5 }
assert_equal "Liquid error: Memory limits exceeded", t.render()
assert t.resource_limits[:reached]
t.resource_limits = { :render_length_limit => 10 }
assert_equal "0123456789", t.render()
assert_not_nil t.resource_limits[:render_length_current]
end
def test_resource_limits_render_score
t = Template.parse("{% for a in (1..10) %} {% for a in (1..10) %} foo {% endfor %} {% endfor %}")
t.resource_limits = { :render_score_limit => 50 }
assert_equal "Liquid error: Memory limits exceeded", t.render()
assert t.resource_limits[:reached]
t = Template.parse("{% for a in (1..100) %} foo {% endfor %}")
t.resource_limits = { :render_score_limit => 50 }
assert_equal "Liquid error: Memory limits exceeded", t.render()
assert t.resource_limits[:reached]
t.resource_limits = { :render_score_limit => 200 }
assert_equal (" foo " * 100), t.render()
assert_not_nil t.resource_limits[:render_score_current]
end
def test_resource_limits_assign_score
t = Template.parse("{% assign foo = 42 %}{% assign bar = 23 %}")
t.resource_limits = { :assign_score_limit => 1 }
assert_equal "Liquid error: Memory limits exceeded", t.render()
assert t.resource_limits[:reached]
t.resource_limits = { :assign_score_limit => 2 }
assert_equal "", t.render()
assert_not_nil t.resource_limits[:assign_score_current]
end
def test_resource_limits_aborts_rendering_after_first_error
t = Template.parse("{% for a in (1..100) %} foo1 {% endfor %} bar {% for a in (1..100) %} foo2 {% endfor %}")
t.resource_limits = { :render_score_limit => 50 }
assert_equal "Liquid error: Memory limits exceeded", t.render()
assert t.resource_limits[:reached]
end
def test_resource_limits_hash_in_template_gets_updated_even_if_no_limits_are_set
t = Template.parse("{% for a in (1..100) %} {% assign foo = 1 %} {% endfor %}")
t.render()
assert t.resource_limits[:assign_score_current] > 0
assert t.resource_limits[:render_score_current] > 0
assert t.resource_limits[:render_length_current] > 0
end
def test_can_use_drop_as_context
t = Template.new
t.registers['lulz'] = 'haha'
drop = TemplateContextDrop.new
assert_equal 'fizzbuzz', t.parse('{{foo}}').render(drop)
assert_equal 'bar', t.parse('{{bar}}').render(drop)
assert_equal 'haha', t.parse("{{baz}}").render(drop)
end
end # TemplateTest end # TemplateTest

View File

@@ -113,6 +113,12 @@ class VariableTest < Test::Unit::TestCase
assert_equal 'hello', var.name assert_equal 'hello', var.name
assert_equal [['things',["greeting: \"world\"","farewell: 'goodbye'"]]], var.filters assert_equal [['things',["greeting: \"world\"","farewell: 'goodbye'"]]], var.filters
end end
def test_lax_filter_argument_parsing
var = Variable.new(%! number_of_comments | pluralize: 'comment': 'comments' !)
assert_equal 'number_of_comments', var.name
assert_equal [['pluralize',["'comment'","'comments'"]]], var.filters
end
end end