module Caboose
PageBarGenerator.class_eval do
def all_records
return model_with_includes.where(where)
end
end
end
module Caboose
class ProductsController < Caboose::ApplicationController
# GET /products || GET /products/:id
def index
# If id exists, is an integer and a product exists with the specified id then get the product
if params[:id] && params[:id].to_i > 0 && Product.exists?(params[:id])
@product = Product.find(params[:id])
render 'product/not_available' and return if @product.status == 'Inactive'
@category = @product.categories.first
@review = Review.new
@reviews = Review.where(:product_id => @product.id).limit(10).order("id DESC") || nil
@logged_in_user = logged_in_user
render 'caboose/products/details' and return
end
# Filter params from url
url_without_params = request.fullpath.split('?').first
# Find the category
category = Category.where(:url => url_without_params).first
# Set category ID
params['category_id'] = category.id
# If this is the top-most category, collect all it's immediate children IDs
params['category_id'] = category.children.collect { |child| child.id } if category.id == 1
# Shove the original category ID into the first position if the param is an array
params['category_id'].unshift(category.id) if params['category_id'].is_a?(Array)
# Otherwise looking at a category or search parameters
@pager = Caboose::Pager.new(params, {
'category_id' => '',
'vendor_id' => '',
'vendor_name' => '',
'vendor_status' => 'Active',
'status' => 'Active',
'variant_status' => 'Active',
'price_gte' => '',
'price_lte' => '',
'alternate_id' => '',
'search_like' => ''
}, {
'model' => 'Caboose::Product',
'sort' => if params[:sort] then params[:sort] else 'store_products.sort_order' end,
'base_url' => url_without_params,
'items_per_page' => 15,
'use_url_params' => false,
'abbreviations' => {
'search_like' => 'title_concat_store_products.alternate_id_concat_vendor_name_concat_category_name_like',
},
'includes' => {
'category_id' => [ 'categories' , 'id' ],
'category_name' => [ 'categories' , 'name' ],
'vendor_id' => [ 'vendor' , 'id' ],
'vendor_name' => [ 'vendor' , 'name' ],
'vendor_status' => [ 'vendor' , 'status' ],
'price_gte' => [ 'variants' , 'price' ],
'price_lte' => [ 'variants' , 'price' ],
'variant_status' => [ 'variants' , 'status' ]
}
})
@sort_options = [
{ :name => 'Default', :value => 'store_products.sort_order' },
{ :name => 'Price (Low to High)', :value => 'store_variants.price ASC' },
{ :name => 'Price (High to Low)', :value => 'store_variants.price DESC' },
{ :name => 'Alphabetical (A-Z)', :value => 'store_products.title ASC' },
{ :name => 'Alphabetical (Z-A)', :value => 'store_products.title DESC' },
]
SearchFilter.delete_all
@filter = SearchFilter.find_from_url(request.fullpath, @pager, ['page'])
@products = @pager.items
@category = if @filter['category_id'] then Category.find(@filter['category_id'].to_i) else nil end
@pager.set_item_count
end
def show
end
# GET /product/info
def info
p = Product.find(params[:id])
render :json => {
:product => p,
:option1_values => p.option1_values,
:option2_values => p.option2_values,
:option3_values => p.option3_values
}
end
#=============================================================================
# API actions
#=============================================================================
# GET /api/products
def api_index
render :json => Product.where(:status => 'Active')
end
# GET /api/products/:id
def api_details
p = Product.where(:id => params[:id]).first
render :json => p ? p : { :error => 'Invalid product ID' }
end
# GET /api/products/:id/variants
def api_variants
p = Product.where(:id => params[:id]).first
render :json => p ? p.variants : { :error => 'Invalid product ID' }
end
#=============================================================================
# Admin actions
#=============================================================================
# GET /admin/products/:id/variants/group
def admin_group_variants
@product = Product.find(params[:id])
return if !user_is_allowed('variants', 'edit')
joins = []
where = []
values = []
if params[:category_ids]
joins << [:category_memberships]
where << 'store_category_memberships.category_id IN (?)'
values << params[:category_ids]
end
if params[:vendor_ids]
joins << [:vendor]
where << 'store_vendors.id IN (?)'
values << params[:vendor_ids]
end
if params[:title]
where << 'LOWER(store_products.title) LIKE ?'
values << "%#{params[:title].downcase}%"
end
# Query for all relevant products
products = values.any? ? Product.joins(joins).where([where.join(' AND ')].concat(values)) : []
# Grab variants for each product
@variants = products.collect { |product| product.variants }.flatten
# Grab all categories; except for "all" and "uncategorized"
@categories = Category.where('parent_id IS NOT NULL AND name IS NOT NULL').order(:url)
# Grab all vendors
@vendors = Vendor.where('name IS NOT NULL').order(:name)
render :layout => 'caboose/admin'
end
# POST /admin/products/:id/variants/add
def admin_add_variants
params[:variant_ids].each do |variant_id|
variant = Variant.find(variant_id)
# Delete current variant product if this is the last variant
# variant.product.update_attribute(:status, 'deleted') if variant.product.variants.where('status != ?', 'deleted').count == 0
# Add reference to new product
# varant.product_id = params[:id]
end
# Iterate over variants and add them to the product
# Remove product that the variants are associated with; UNLESS it's the current product
redirect_to "/admin/products/#{params[:id]}/variants"
end
# POST /admin/products/:id/varaints/add-multiple
def admin_add_multiple_variants
product = Product.find(params[:id])
params[:variants_csv].split("\r\n").each do |variant|
row = variant.split(',')
render :json => { :success => false, :error => "Quantity is not defined for variant: #{row[0].strip}" } and return if row[1].nil?
render :json => { :success => false, :error => "Price is not defined for variant: #{row[0].strip}" } and return if row[2].nil?
attributes = {
:alternate_id => row[0].strip,
:quantity_in_stock => row[1].strip.to_i,
:price => '%.2f' % row[2].strip.to_f,
:status => 'Active'
}
if product.option1 && row[3].nil?
render :json => { :success => false, :error => "#{product.option1} not defined for variant: #{attributes[:alternate_id]}" } and return
elsif product.option1
attributes[:option1] = row[3].strip
end
if product.option2 && row[4].nil?
render :json => { :success => false, :error => "#{product.option2} not defined for variant: #{attributes[:alternate_id]}" } and return
elsif product.option2
attributes[:option2] = row[4].strip
end
if product.option3 && row[5].nil?
render :json => { :success => false, :error => "#{product.option3} not defined for variant: #{attributes[:alternate_id]}" } and return
elsif product.option3
attributes[:option3] = row[5].strip
end
if product.variants.find_by_alternate_id(attributes[:alternate_id])
product.variants.find_by_alternate_id(attributes[:alternate_id]).update_attributes(attributes)
else
Variant.create(attributes.merge(:product_id => product.id))
end
end
render :json => { :success => true }
end
# POST /admin//products/:id/variants/remove
def admin_remove_variants
params[:variant_ids].each do |variant_id|
variant = Variant.find(variant_id)
# variant.update_attribute(:status, 'deleted')
# variant.product.update_attribute(:status, 'deleted') if variant.product.variants.where('status != ?', 'deleted').count == 0
end
# Remove passed variants
# redirect_to "/admin/products/#{params[:id]}/variants/group"
render :json => true
end
# GET /admin/products/update-vendor-status/:id
def admin_update_vendor_status
vendor = Vendor.find(params[:id])
vendor.status = params[:status]
render :json => vendor.save
end
# GET /admin/products
def admin_index
return if !user_is_allowed('products', 'view')
# Temporary patch for vendor name sorting; Fix this
params[:sort] = 'store_vendors.name' if params[:sort] == 'vendor'
@gen = Caboose::PageBarGenerator.new(params, {
'vendor_name' => '',
'search_like' => '',
'price' => params[:filters] && params[:filters][:missing_prices] ? 0 : ''
}, {
'model' => 'Caboose::Product',
'sort' => 'title',
'desc' => false,
'base_url' => '/admin/products',
'items_per_page' => 25,
'use_url_params' => false,
'abbreviations' => {
'search_like' => 'store_products.title_concat_vendor_name_like'
},
'includes' => {
'vendor_name' => [ 'vendor' , 'name' ],
'price' => [ 'variants' , 'price' ]
}
})
# Make a copy of all the items; so it can be filtered more
@all_products = @gen.all_records
# Apply any extra filters
if params[:filters]
@all_products = @all_products.includes(:product_images).where('store_product_images.id IS NULL') if params[:filters][:missing_images]
@all_products = @all_products.where('vendor_id IS NULL') if params[:filters][:no_vendor]
end
# Get the correct page of the results
@products = @all_products.limit(@gen.limit).offset(@gen.offset)
render :layout => 'caboose/admin'
end
# GET /admin/products/json
def admin_json
return if !user_is_allowed('products', 'view')
# Temporary patch for vendor name sorting; Fix this
params[:sort] = 'store_vendors.name' if params[:sort] == 'vendor'
pager = Caboose::PageBarGenerator.new(params, {
'vendor_name' => '',
'search_like' => '',
'price' => params[:filters] && params[:filters][:missing_prices] ? 0 : ''
}, {
'model' => 'Caboose::Product',
'sort' => 'title',
'desc' => false,
'base_url' => '/admin/products',
'items_per_page' => 25,
'use_url_params' => false,
'abbreviations' => {
'search_like' => 'store_products.title_concat_vendor_name_like'
},
'includes' => {
'vendor_name' => [ 'vendor' , 'name' ],
'price' => [ 'variants' , 'price' ]
}
})
render :json => {
:pager => pager,
:models => pager.items
}
end
# GET /admin/products/add-upcs - TODO remove this; it's a temporary thing for woods-n-water
def admin_add_upcs
params[:vendor_id] if params[:vendor_id] and params[:vendor_id].empty?
conditions = if params[:vendor_id]
"store_variants.alternate_id IS NULL and store_vendors.id = #{params[:vendor_id]}"
else
"store_variants.alternate_id IS NULL"
end
@products = Product.all(
:include => [:variants, :vendor],
:conditions => conditions
)
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/json
def admin_json_single
p = Product.find(params[:id])
render :json => p
end
# GET /admin/products/:id/general
def admin_edit_general
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/description
def admin_edit_description
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/variants
# GET /admin/products/:id/variants/:variant_id
def admin_edit_variants
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
if @product.variants.nil? || @product.variants.count == 0
v = Variant.new
v.option1 = @product.default1 if @product.option1
v.option2 = @product.default2 if @product.option2
v.option3 = @product.default3 if @product.option3
v.status = 'Active'
@product.variants = [v]
@product.save
end
@variant = params[:variant_id] ? Variant.find(params[:variant_id]) : @product.variants[0]
session['variant_cols'] = self.default_variant_cols if session['variant_cols'].nil?
@cols = session['variant_cols']
@highlight_variant_id = params[:highlight] ? params[:highlight].to_i : nil
if @product.options.nil? || @product.options.count == 0
render 'caboose/products/admin_edit_variants_single', :layout => 'caboose/admin'
else
render 'caboose/products/admin_edit_variants', :layout => 'caboose/admin'
end
end
# GET /admin/products/:id/variants/json
def admin_variants_json
render :json => false if !user_is_allowed('products', 'edit')
p = Product.find(params[:id])
render :json => p.variants
end
# GET /admin/products/:id/variant-cols
def admin_edit_variant_columns
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
session['variant_cols'] = self.default_variant_cols if session['variant_cols'].nil?
@cols = session['variant_cols']
render :layout => 'caboose/admin'
end
# PUT /admin/products/:id/variant-cols
def admin_update_variant_columns
return if !user_is_allowed('products', 'edit')
session['variant_cols'] = self.default_variant_cols if session['variant_cols'].nil?
resp = Caboose::StdClass.new({'attributes' => {}})
product = Product.find(params[:id])
save = true
params.each do |name,value|
value = ActiveRecord::ConnectionAdapters::Column.value_to_boolean(value)
case name
when 'option1' ,
'option2' ,
'option3' ,
'status' ,
'alternate_id' ,
'sku' ,
'barcode' ,
'price' ,
'quantity_in_stock' ,
'weight' ,
'length' ,
'width' ,
'height' ,
'cylinder' ,
'requires_shipping' ,
'allow_backorder' ,
'taxable'
session['variant_cols'][name] = value
end
end
resp.success = save && product.save
render :json => resp
end
def default_variant_cols
return {
'option1' => true,
'option2' => true,
'option3' => true,
'status' => true,
'alternate_id' => true,
'sku' => true,
'barcode' => false,
'price' => true,
'quantity' => true,
'weight' => false,
'length' => false,
'width' => false,
'height' => false,
'cylinder' => false,
'requires_shipping' => false,
'allow_backorder' => false,
'taxable' => false
}
end
# GET /admin/products/:id/options
def admin_edit_options
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/categories
def admin_edit_categories
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
@top_categories = Category.where(:parent_id => 1).reorder('name').all
@selected_ids = @product.categories.collect{ |cat| cat.id }
render :layout => 'caboose/admin'
end
# POST /admin/products/:id/categories
def admin_add_to_category
return if !user_is_allowed('products', 'edit')
cat_id = params[:category_id]
product_id = params[:id]
if !CategoryMembership.exists?(:category_id => cat_id, :product_id => product_id)
CategoryMembership.create(:category_id => cat_id, :product_id => product_id)
end
render :json => true
end
# DELETE /admin/products/:id/categories/:category_id
def admin_remove_from_category
return if !user_is_allowed('products', 'edit')
cat_id = params[:category_id]
product_id = params[:id]
if CategoryMembership.exists?(:category_id => cat_id, :product_id => product_id)
CategoryMembership.where(:category_id => cat_id, :product_id => product_id).destroy_all
end
render :json => true
end
# GET /admin/products/:id/images
def admin_edit_images
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# POST /admin/products/:id/images
def admin_add_image
return if !user_is_allowed('products', 'edit')
product_id = params[:id]
if (params[:new_image].nil?)
render :text => ""
else
img = ProductImage.new
img.product_id = product_id
img.image = params[:new_image]
img.square_offset_x = 0
img.square_offset_y = 0
img.square_scale_factor = 1.00
img.save
render :text => ""
end
end
# GET /admin/products/:id/collections
def admin_edit_collections
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/seo
def admin_edit_seo
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/variants/sort-order
def admin_edit_variant_sort_order
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# GET /admin/products/:id/delete
def admin_delete_form
return if !user_is_allowed('products', 'edit')
@product = Product.find(params[:id])
render :layout => 'caboose/admin'
end
# PUT /admin/products/:id
def admin_update
return if !user_is_allowed('products', 'edit')
resp = Caboose::StdClass.new({'attributes' => {}})
product = Product.find(params[:id])
save = true
params.each do |name,value|
case name
when 'alternate_id' then product.alternate_id = value
when 'title' then product.title = value
when 'caption' then product.caption = value
when 'featured' then product.featured = value
when 'description' then product.description = value
when 'vendor_id' then product.vendor_id = value
when 'handle' then product.handle = value
when 'seo_title' then product.seo_title = value
when 'seo_description' then product.seo_description = value
when 'status' then product.status = value
when 'category_id' then product.toggle_category(value[0], value[1])
when 'stackable_group_id' then product.stackable_group_id = value
when 'option1' then product.option1 = value
when 'option2' then product.option2 = value
when 'option3' then product.option3 = value
when 'default1'
product.default1 = value
Variant.where(:product_id => product.id, :option1 => nil).each do |p|
p.option1 = value
p.save
end
when 'default2'
product.default2 = value
Variant.where(:product_id => product.id, :option2 => nil).each do |p|
p.option2 = value
p.save
end
when 'default3'
product.default3 = value
Variant.where(:product_id => product.id, :option3 => nil).each do |p|
p.option3 = value
p.save
end
when 'date_available'
if value.strip.length == 0
product.date_available = nil
else
begin
product.date_available = DateTime.parse(value)
rescue
resp.error = "Invalid date"
save = false
end
end
end
end
resp.success = save && product.save
render :json => resp
end
# GET /admin/products/new
def admin_new
return if !user_is_allowed('products', 'add')
render :layout => 'caboose/admin'
end
# POST /admin/products
def admin_add
return if !user_is_allowed('products', 'add')
resp = Caboose::StdClass.new(
:error => nil,
:redirect => nil
)
title = params[:title]
if title.length == 0
resp.error = "The title cannot be empty."
else
p = Product.new(:title => title)
p.save
resp.redirect = "/admin/products/#{p.id}/general"
end
render :json => resp
end
# DELETE /admin/products/:id
def admin_delete
return if !user_is_allowed('products', 'delete')
p = Product.find(params[:id]).destroy
p.status = 'Deleted'
p.save
render :json => Caboose::StdClass.new({
:redirect => '/admin/products'
})
end
# GET /products/status-options
def admin_status_options
arr = ['Active', 'Inactive', 'Deleted']
options = []
arr.each do |status|
options << {
:value => status,
:text => status
}
end
render :json => options
end
# GET /products/stackable-group-options
def admin_stackable_group_options
arr = ['Active', 'Inactive', 'Deleted']
options = []
arr.each do |status|
options << {
:value => status,
:text => status
}
end
render :json => options
end
# GET /admin/products/combine
def admin_combine_select_products
end
# GET /admin/products/combine-step2
def admin_combine_assign_title
end
# POST /admin/products/combine
def admin_combine
product_ids = params[:product_ids]
p = Product.new
p.title = params[:title]
p.description = params[:description]
p.option1 = params[:option1]
p.option2 = params[:option2]
p.option3 = params[:option3]
p.default1 = params[:default1]
p.default2 = params[:default2]
p.default3 = params[:default3]
p.status = 'Active'
p.save
product_ids.each do |pid|
p = Product.find(pid)
p.variants.each do |v|
end
end
end
# PUT /admin/products/:id/update-vendor
def admin_update_vendor
render :json => { :success => Product.find(params[:id]).update_attribute(:vendor_id, params[:vendor_id]) }
end
# GET /admin/products/sort
def admin_sort
@products = Product.active
@vendors = Vendor.active
@categories = Category.all
render :layout => 'caboose/admin'
end
# PUT /admin/products/update-sort-order
def admin_update_sort_order
params[:product_ids].each_with_index do |product_id, index|
Product.find(product_id.to_i).update_attribute(:sort_order, index)
end
render :json => { :success => true }
end
# PUT /admin/products/:id/variants/option1-sort-order
def admin_update_variant_option1_sort_order
product_id = params[:id]
params[:values].each_with_index do |value, i|
Variant.where(:product_id => product_id, :option1 => value).all.each do |v|
v.update_attribute(:option1_sort_order, i)
end
end
render :json => { :success => true }
end
# PUT /admin/products/:id/variants/option1-sort-order
def admin_update_variant_option2_sort_order
product_id = params[:id]
params[:values].each_with_index do |value, i|
Variant.where(:product_id => product_id, :option2 => value).all.each do |v|
v.update_attribute(:option2_sort_order, i)
end
end
render :json => { :success => true }
end
# PUT /admin/products/:id/variants/option1-sort-order
def admin_update_variant_option3_sort_order
product_id = params[:id]
params[:values].each_with_index do |value, i|
Variant.where(:product_id => product_id, :option3 => value).all.each do |v|
v.update_attribute(:option3_sort_order, i)
end
end
render :json => { :success => true }
end
end
end