module Spree
module Admin
module NavigationHelper
# Makes an admin navigation tab (
tag) that links to a routing resource under /admin.
# The arguments should be a list of symbolized controller names that will cause this tab to
# be highlighted, with the first being the name of the resouce to link (uses URL helpers).
#
# Option hash may follow. Valid options are
# * :label to override link text, otherwise based on the first resource name (translated)
# * :route to override automatically determining the default route
# * :match_path as an alternative way to control when the tab is active, /products would
# match /admin/products, /admin/products/5/variants etc. Can be a String or a Regexp.
# Controller names are ignored if :match_path is provided.
#
# Example:
# # Link to /admin/orders, also highlight tab for ProductsController and ShipmentsController
# tab :orders, :products, :shipments
def tab(*args)
options = { label: args.first.to_s }
# Return if resource is found and user is not allowed to :admin
return '' if (klass = klass_for(options[:label])) && cannot?(:admin, klass)
options = options.merge(args.pop) if args.last.is_a?(Hash)
options[:route] ||= "admin_#{args.first}"
destination_url = options[:url] || spree.send("#{options[:route]}_path")
titleized_label = Spree.t(options[:label], default: options[:label], scope: [:admin, :tab]).titleize
css_classes = ['sidebar-menu-item']
link = if options[:icon]
link_to_with_icon(options[:icon], titleized_label, destination_url)
else
link_to(titleized_label, destination_url)
end
selected = if options[:match_path].is_a? Regexp
request.fullpath =~ options[:match_path]
elsif options[:match_path]
request.fullpath.starts_with?("#{spree.admin_path}#{options[:match_path]}")
else
args.include?(controller.controller_name.to_sym)
end
css_classes << 'selected' if selected
css_classes << options[:css_class] if options[:css_class]
content_tag('li', link, class: css_classes.join(' '))
end
# Single main menu item
def main_menu_item(text, url: nil, icon: nil)
link_to url, 'data-toggle': 'collapse', 'data-parent': '#sidebar' do
content_tag(:span, nil, class: "icon icon-#{icon}") +
content_tag(:span, " #{text}", class: 'text') +
content_tag(:span, nil, class: 'icon icon-chevron-left pull-right')
end
end
# Main menu tree menu
def main_menu_tree(text, icon: nil, sub_menu: nil, url: '#')
content_tag :li, class: 'sidebar-menu-item' do
main_menu_item(text, url: url, icon: icon) +
render(partial: "spree/admin/shared/sub_menu/#{sub_menu}")
end
end
# the per_page_dropdown is used on index pages like orders, products, promotions etc.
# this method generates the select_tag
def per_page_dropdown
# there is a config setting for admin_products_per_page, only for the orders page
if @products && per_page_default = Spree::Config.admin_products_per_page
per_page_options = []
5.times do |amount|
per_page_options << (amount + 1) * Spree::Config.admin_products_per_page
end
else
per_page_default = Spree::Config.admin_orders_per_page
per_page_options = %w{15 30 45 60}
end
selected_option = params[:per_page].try(:to_i) || per_page_default
select_tag(:per_page,
options_for_select(per_page_options, selected_option),
class: "form-control pull-right js-per-page-select per-page-selected-#{selected_option}")
end
# helper method to create proper url to apply per page filtering
# fixes https://github.com/spree/spree/issues/6888
def per_page_dropdown_params(args = nil)
args = params.permit!.to_h.clone
args.delete(:page)
args.delete(:per_page)
args
end
# finds class for a given symbol / string
#
# Example :
# :products returns Spree::Product
# :my_products returns MyProduct if MyProduct is defined
# :my_products returns My::Product if My::Product is defined
# if cannot constantize it returns nil
# This will allow us to use cancan abilities on tab
def klass_for(name)
model_name = name.to_s
["Spree::#{model_name.classify}", model_name.classify, model_name.tr('_', '/').classify].find(&:safe_constantize).try(:safe_constantize)
end
def link_to_clone(resource, options = {})
options[:data] = { action: 'clone', 'original-title': Spree.t(:clone) }
options[:class] = 'btn btn-primary btn-sm with-tip'
options[:method] = :post
options[:icon] = :clone
button_link_to '', clone_object_url(resource), options
end
def link_to_edit(resource, options = {})
url = options[:url] || edit_object_url(resource)
options[:data] = { action: 'edit' }
options[:class] = 'btn btn-primary btn-sm'
link_to_with_icon('edit', Spree.t(:edit), url, options)
end
def link_to_edit_url(url, options = {})
options[:data] = { action: 'edit' }
options[:class] = 'btn btn-primary btn-sm'
link_to_with_icon('edit', Spree.t(:edit), url, options)
end
def link_to_delete(resource, options = {})
url = options[:url] || object_url(resource)
name = options[:name] || Spree.t(:delete)
options[:class] = 'btn btn-danger btn-sm delete-resource'
options[:data] = { confirm: Spree.t(:are_you_sure), action: 'remove' }
link_to_with_icon 'delete', name, url, options
end
def link_to_with_icon(icon_name, text, url, options = {})
options[:class] = (options[:class].to_s + " icon-link with-tip action-#{icon_name}").strip
options[:class] += ' no-text' if options[:no_text]
options[:title] = text if options[:no_text]
text = options[:no_text] ? '' : content_tag(:span, text, class: 'text')
options.delete(:no_text)
if icon_name
icon = content_tag(:span, '', class: "icon icon-#{icon_name}")
text.insert(0, icon + ' ')
end
link_to(text.html_safe, url, options)
end
def icon(icon_name)
icon_name ? content_tag(:i, '', class: icon_name) : ''
end
# Override: Add disable_with option to prevent multiple request on consecutive clicks
def button(text, icon_name = nil, button_type = 'submit', options = {})
if icon_name
icon = content_tag(:span, '', class: "icon icon-#{icon_name}")
text.insert(0, icon + ' ')
end
button_tag(text.html_safe, options.merge(type: button_type, class: "btn btn-primary #{options[:class]}", 'data-disable-with' => "#{Spree.t(:saving)}..."))
end
def button_link_to(text, url, html_options = {})
if html_options[:method] &&
!html_options[:method].to_s.casecmp('get').zero? &&
!html_options[:remote]
form_tag(url, method: html_options.delete(:method), class: 'display-inline') do
button(text, html_options.delete(:icon), nil, html_options)
end
else
if html_options['data-update'].nil? && html_options[:remote]
object_name, action = url.split('/')[-2..-1]
html_options['data-update'] = [action, object_name.singularize].join('_')
end
html_options.delete('data-update') unless html_options['data-update']
html_options[:class] = html_options[:class] ? "btn #{html_options[:class]}" : 'btn btn-default'
if html_options[:icon]
icon = content_tag(:span, '', class: "icon icon-#{html_options[:icon]}")
text.insert(0, icon + ' ')
end
link_to(text.html_safe, url, html_options)
end
end
def configurations_sidebar_menu_item(link_text, url, options = {})
is_selected = url.ends_with?(controller.controller_name) ||
url.ends_with?("#{controller.controller_name}/edit") ||
url.ends_with?("#{controller.controller_name.singularize}/edit")
options[:class] = 'sidebar-menu-item'
options[:class] << ' selected' if is_selected
content_tag(:li, options) do
link_to(link_text, url)
end
end
def main_part_classes
if cookies['sidebar-minimized'] == 'true'
'col-xs-12 sidebar-collapsed'
else
'col-xs-9 col-xs-offset-3 col-md-10 col-md-offset-2'
end
end
def main_sidebar_classes
if cookies['sidebar-minimized'] == 'true'
'col-xs-3 col-md-2 hidden-xs sidebar'
else
'col-xs-3 col-md-2 sidebar'
end
end
def wrapper_classes
'sidebar-minimized' if cookies['sidebar-minimized'] == 'true'
end
end
end
end