Introduction

This tutorial continues where we left off in the Getting Started tutorial. Now that we have a basic Spree store up and running, let’s spend some time customizing it. The easiest way to do this is by using Spree extensions.

What is a Spree Extension?

Extensions are the primary mechanism for customizing a Spree site. They provide a convenient mechanism for Spree developers to share reusable code with one another. Even if you do not plan on sharing your extensions with the community, they can still be a useful way to reuse code within your organization. Extensions are also a convenient mechanism for organizing and isolating discrete chunks of functionality.

Finding Useful Spree Extensions in the Extension Registry

The Spree Extension Registry is a searchable collection of Spree Extensions written and maintained by members of the Spree Community. If you need to extend your Spree application’s functionality, be sure to have a look in the Extension Registry first; you may find an extension that either implements what you need or provides a good starting point for your own implementation. If you write an extension and it might be useful to others, publish it in the registry and people will be able to find it and contribute as well.

Installing an Extension

We are going to be adding the spree_fancy extension to our store. SpreeFancy is a theme so it only changes the look and feel of the application. Extensions can also add models, controllers, and views to create new functionality, but spree_fancy is intended as a starting point to show how a barebones Spree application can be easily modified to give a nice look and feel. As a special bonus it’s fully responsive and looks good on mobile devices as well as on larger screens.

There are three steps we need to take to install spree_fancy.

First, we need to add the gem to the bottom of our Gemfile:

gem 'spree_fancy', :git => 'git://github.com/spree/spree_fancy.git', :branch => '2-1-stable'


Note that if you are using the edge version of spree, you should omit the branch parameter to get the latest version of spree_fancy. Alternatively, you should select the version of spree_fancy that corresponds with your version of spree.

If you are using a 2.1.x version of Spree, the above line will work fine. If you’re using a 2.0.x version of Spree, you’ll need to change the “branch” option to point to the “2-0-stable” branch. If you’re using the “master” branch of Spree, change the “branch” argument for “spree_fancy” to be “master” as well.

Now, let’s install the gem via Bundler with the following command:

$ bundle install

Finally, let’s copy over the required migrations and assets from the extension with the following command:

$ bundle exec rails g spree_fancy:install

Answer yes when prompted to run migrations.

When the last command is done running, you can start your application again and navigate to http://localhost:3000 to see our brand new theme.

Creating an Extension

Getting Started

Let’s build a simple extension. Suppose we want the ability to mark certain products as being on sale. We’d like to be able to set a sale price on a product and show products that are on sale on a separate products page. This is a great example of how an extension can be used to build on the solid Spree foundation.

So let’s start by generating the extension. Run the following command from a directory of your choice outside of our Spree application:

$ spree extension simple_sales

This creates a spree_simple_sales directory with several additional files and directories. After generating the extension make sure you change to its directory:

$ cd spree_simple_sales

Adding a Sale Price to Variants

The first thing we need to do is create a migration that adds a sale_price column to variants.

We can do this with the following command:

rails g migration add_sale_price_to_spree_variants sale_price:decimal

**************** TODO ****************

Make above generator actually work in extension directories

**************************************

Because we are dealing with prices, we need to now edit the generated migration to ensure the correct precision and scale. Edit the file db/migrate/XXXXXXXXXXX_add_sale_price_to_spree_variants.rb so that it contains the following:

class AddSalePriceToSpreeVariants < ActiveRecord::Migration
  def change
    add_column :spree_variants, :sale_price, :decimal, :precision => 8, :scale => 2
  end
end

Adding Our Extension to the Spree Application

Before we continue development of our extension, let’s add it to the Spree application we created in the last tutorial. This will allow us to see how the extension works with an actual Spree store while we develop it.

Within the mystore application directory, add the following line to the bottom of our Gemfile:

gem 'spree_simple_sales', :path => '../spree_simple_sales'

You may have to adjust the path somewhat depending on where you created the extension. You want this to be the path relative to the location of the mystore application.

Once you have added the gem, it’s time to bundle:

$ bundle install

Finally, let’s run the spree_simple_sales install generator to copy over the migration we just created (answer yes if prompted to run migrations):

# context: Your Spree store's app root (i.e. Rails.root); not the extension's root path.
$ rails g spree_simple_sales:install

Adding a Controller Action to HomeController

Now we need to extend Spree::HomeController and add an action that selects “on sale” products.

Make sure you are in the spree_simple_sales root directory and run the following command to create the directory structure for our controller decorator:

$ mkdir -p app/controllers/spree

Next, create a new file in the directory we just created called home_controller_decorator.rb and add the following content to it:

module Spree
  HomeController.class_eval do
    def sale
      @products = Product.joins(:variants_including_master).where('spree_variants.sale_price is not null').uniq
    end
  end
end

This will select just the products that have a variant with a sale_price set.

We also need to add a route to this action in our config/routes.rb file. Let’s do this now. Update the routes file to contain the following:

Spree::Core::Engine.routes.draw do
  get "/sale" => "home#sale"
end

Viewing On Sale Products

Setting the Sale Price for a Variant

Now that our variants have the attribute sale_price available to them, let’s update the sample data so we have at least one product that is on sale in our application. We will need to do this in the rails console for the time being, as we have no admin interface to set sale prices for variants. We will be adding this functionality in the next tutorial in this series, Deface overrides.

So, in order to do this, first open up the rails console:

$ rails console

Now, follow the steps I take in selecting a product and updating its master variant to have a sale price. Note, you may not be editing the exact same product as I am, but this is not important. We just need one “on sale” product to display on the sales page.

> product = Spree::Product.first
=> #<Spree::Product id: 107377505, name: "Spree Bag", description: "Lorem ipsum dolor sit amet, consectetuer adipiscing...", available_on: "2013-02-13 18:30:16", deleted_at: nil, permalink: "spree-bag", meta_description: nil, meta_keywords: nil, tax_category_id: 25484906, shipping_category_id: nil, count_on_hand: 10, created_at: "2013-02-13 18:30:16", updated_at: "2013-02-13 18:30:16", on_demand: false>

> variant = product.master
=> #<Spree::Variant id: 833839126, sku: "SPR-00012", weight: nil, height: nil, width: nil, depth: nil, deleted_at: nil, is_master: true, product_id: 107377505, count_on_hand: 10, cost_price: #<BigDecimal:7f8dda5eebf0,'0.21E2',9(36)>, position: nil, lock_version: 0, on_demand: false, cost_currency: nil, sale_price: nil>

> variant.sale_price = 8.00
=> 8.0

> variant.save
=> true

Creating a View

Now we have at least one product in our database that is on sale. Let’s create a view to display these products.

First, create the required views directory with the following command:

$ mkdir -p app/views/spree/home

Next, create the file app/views/spree/home/sale.html.erb and add the following content to it:

<div data-hook="homepage_products">
  <%= render 'spree/shared/products', :products => @products %>
</div>

If you navigate to http://localhost:3000/sale you should now see the product(s) listed that we set a sale_price on earlier in the tutorial. However, if you look at the price, you’ll notice that it’s not actually displaying the correct price. This is easy enough to fix and we will cover that in the next section.

Decorating Variants

Let’s fix our extension so that it uses the sale_price when it is present.

First, create the required directory structure for our new decorator:

$ mkdir -p app/models/spree

Next, create the file app/models/spree/variant_decorator.rb and add the following content to it:

module Spree
  Variant.class_eval do
    alias_method :orig_price_in, :price_in
    def price_in(currency)
      return orig_price_in(currency) unless sale_price.present?
      Spree::Price.new(:variant_id => self.id, :amount => self.sale_price, :currency => currency)
    end
  end
end

Here we alias the original method price_in to orig_price_in and override it. If there is a sale_price present on the product’s master variant, we return that price. Otherwise, we call the original implementation of price_in.

Testing Our Decorator

It’s always a good idea to test your code. We should be extra careful to write tests for our Variant decorator since we are modifying core Spree functionality. Let’s write a couple of simple unit tests for variant_decorator.rb

Generating the Test App

An extension is not a full Rails application, so we need something to test our extension against. By running the Spree test_app rake task, we can generate a barebones Spree application within our spec directory to run our tests against.

We can do this with the following command from the root directory of our extension:

$ bundle exec rake test_app

After this command completes, you should be able to run rspec and see the following output:

No examples found.

Finished in 0.00005 seconds
0 examples, 0 failures

Great! We’re ready to start adding some tests. Let’s replicate the extension’s directory structure in our spec directory by running the following command

$ mkdir -p spec/models/spree

Now, let’s create a new file in this directory called variant_decorator_spec.rb and add the following tests to it:

require 'spec_helper'

describe Spree::Variant do
  describe "#price_in" do
    it "returns the sale price if it is present" do
      variant = create(:variant, :sale_price => 8.00)
      expected = Spree::Price.new(:variant_id => variant.id, :currency => "USD", :amount => variant.sale_price)

      result = variant.price_in("USD")

      result.variant_id.should == expected.variant_id
      result.amount.to_f.should == expected.amount.to_f
      result.currency.should == expected.currency
    end

    it "returns the normal price if it is not on sale" do
      variant = create(:variant, :price => 15.00)
      expected = Spree::Price.new(:variant_id => variant.id, :currency => "USD", :amount => variant.price)

      result = variant.price_in("USD")

      result.variant_id.should == expected.variant_id
      result.amount.to_f.should == expected.amount.to_f
      result.currency.should == expected.currency
    end
  end
end

These specs test that the price_in method we overrode in our VariantDecorator returns the correct price both when the sale price is present and when it is not.

Versioning your extension

Different versions of Spree may act differently with your extension. It’s advisable to keep different branches of your extension actively maintained for the different branches of Spree so that your extension will work with those different versions.

It’s advisable that your extension follows the same versioning pattern as Spree itself. If your extension is compatible with Spree 2.0.x, then create a 2-0-stable branch on your extension and advise people to use that branch for your extension. If it’s only compatible with 1.3.x, then create a 1-3-stable branch and advise the use of that branch.

Having a consistent branching naming scheme across Spree and its extensions will reduce confusion in the long run.

Summary

In this tutorial you learned how to both install extensions and create your own. A lot of core Spree development concepts were covered and you gained exposure to some of the Spree internals.

In the next part of this tutorial series, we will cover Deface overrides and look at ways to improve our current extension.