module Spree module Admin module NavigationHelper # Make an admin tab that coveres one or more resources supplied by symbols # 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. 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]) and cannot?(:admin, klass) if args.last.is_a?(Hash) options = options.merge(args.pop) end 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'] if options[:icon] link = link_to_with_icon(options[:icon], titleized_label, destination_url) else link = 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 if options[:css_class] css_classes << options[:css_class] end 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 # sidebar are used on order edit, product edit, user overview etc. # this link is shown so a user can collapse the sidebar def collapse_sidebar_link content_tag :div, class: "collapse-sidebar" do link_to "javascript:;", class: "js-collapse-sidebar" do content_tag(:span, nil, class: "icon icon-chevron-right") + content_tag(:span, "Collapse sidebar", class: "text") end 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 = 15 per_page_options = %w{5 15 30 45 60} end select_tag(:per_page, options_for_select(per_page_options, params['per_page'] || per_page_default), { id: "js-per-page-select", class: "form-control pull-right" }) 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.gsub('_', '/').classify].find do |t| t.safe_constantize end.try(:safe_constantize) end def link_to_clone(resource, options={}) options[:data] = { action: 'clone' } options[:class] = "btn btn-primary btn-sm" link_to_with_icon('clone', Spree.t(:clone), clone_object_url(resource), options) end def link_to_new(resource) options[:data] = { action: 'new' } options[:class] = "btn btn-success btn-sm" link_to_with_icon('plus', Spree.t(:new), edit_object_url(resource)) 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] ? '' : raw("#{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 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]}")) end def button_link_to(text, url, html_options = {}) if (html_options[:method] && html_options[:method].to_s.downcase != 'get' && !html_options[:remote]) form_tag(url, method: html_options.delete(:method)) 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_menu_item(link_text, url, description = '') %(