module Alchemy module Admin # This module contains helper methods for rendering overlay windows, toolbar buttons and confirmation windows. # # The most important helpers for module developers are: # # * toolbar # * toolbar_button # * link_to_overlay_window # * link_to_confirmation_window # module BaseHelper # This helper renders the link for an overlay window. # # We use this for our fancy modal overlay windows in the Alchemy cockpit. # # === Options: # # :size [String] # String with format of "WidthxHeight". I.E. ("420x280") # :title [String] # Text for the overlay title bar. # :overflow [Boolean] # Should the dialog have overlapping content. If not, it shows scrollbars. Good for select boxes. Default false. # :resizable [Boolean] # Is the dialog window resizable? Default false. # :modal [Boolean] # Show as modal window. Default true. # def link_to_overlay_window(content, url, options={}, html_options={}) default_options = { :modal => true, :overflow => false, :resizable => false } options = default_options.merge(options) link_to_function( content, "Alchemy.openWindow( \'#{url}\', \'#{options[:title]}\', \'#{options[:size] ? options[:size].split('x')[0].to_s : 'auto'}\', \'#{options[:size] ? options[:size].split('x')[1].to_s : 'auto'}\', #{options[:resizable]}, #{options[:modal]}, #{options[:overflow]} )", html_options ) end # (internal) Used for rendering the folder link in +Admin::Pages#index+ sitemap. def sitemapFolderLink(page) return '' if page.level == 1 if page.folded?(current_user.id) css_class = 'folded' title = t('Show childpages') else css_class = 'collapsed' title = t('Hide childpages') end link_to( '', alchemy.fold_admin_page_path(page), :remote => true, :method => :post, :class => "page_folder #{css_class}", :title => title, :id => "fold_button_#{page.id}" ) end # Used for language selector in Alchemy cockpit sitemap. So the user can select the language branche of the page. def language_codes_for_select configuration(:languages).collect { |language| language[:language_code] } end # Used for translations selector in Alchemy cockpit user settings. def translations_for_select Alchemy::I18n.available_locales.map do |locale| [t(locale, :scope => :translations), locale] end end # Used by Alchemy to display a javascript driven filter for lists in the Alchemy cockpit. def js_filter_field options = {} default_options = { :class => "thin_border js_filter_field", :onkeyup => "Alchemy.ListFilter('#contact_list li')", :id => "search_field" } options = default_options.merge(options) options[:onkeyup] << ";jQuery('#search_field').val().length >= 1 ? jQuery('.js_filter_field_clear').show() : jQuery('.js_filter_field_clear').hide();" filter_field = "
" filter_field << text_field_tag("filter", '', options) filter_field << content_tag('span', '', :class => 'icon search') filter_field << link_to_function( "", "jQuery('##{options[:id]}').val('');#{options[:onkeyup]}", :class => "js_filter_field_clear", :style => "display:none", :title => t("click_to_show_all") ) filter_field << "" filter_field << "
" filter_field.html_safe end # Returns a link that opens a modal confirmation window. # # === Parameters: # # 1. The content inside the tag # 2. The message that is displayed in the overlay window # 3. The url that gets opened after confirmation (Note: This is an Ajax request with a method of DELETE!) # 4. html options get passed to the link # # === Example: # # <%= link_to_confirmation_window('delete', 'Do you really want to delete this comment?', '/admin/comments/1') %> # def link_to_confirmation_window(link_string = "", message = "", url = "", html_options = {}) title = t("please_confirm") ok_lable = t("Yes") cancel_lable = t("No") link_to_function( link_string, "Alchemy.confirmToDeleteWindow('#{url}', '#{title}', '#{message}', '#{ok_lable}', '#{cancel_lable}');", html_options ) end # Returns an Array build for passing it to the options_for_select helper inside an essence editor partial. # Usefull for the select_values options from the render_essence_editor helpers. # # == Options: # # :from_page [String, Page] # Return only elements from this page. You can either pass a Page instance, or a page_layout name # :elements_with_name [Array, String] # Return only elements with this name(s). # def elements_for_essence_editor_select(options={}) defaults = { :from_page => nil, :elements_with_name => nil, :prompt => t('Please choose') } options = defaults.merge(options) if options[:from_page] page = options[:from_page].is_a?(String) ? Page.find_by_page_layout(options[:from_page]) : options[:from_page] end if page elements = options[:elements_with_name].blank? ? page.elements.find_all_by_public(true) : page.elements.find_all_by_public_and_name(true, options[:elements_with_name]) else elements = options[:elements_with_name].blank? ? Element.find_all_by_public(true) : Element.find_all_by_public_and_name(true, options[:elements_with_name]) end select_options = [[options[:prompt], ""]] elements.each do |e| select_options << [e.display_name_with_preview_text, e.id.to_s] end select_options end # Returns all public pages found in the database as an Array suitable or the Rails +select_tag+ helper. # # * You can pass a collection of pages so it only returns these pages and does not query the database. # * Pass a +Page#name+ or +Page#id+ as second parameter to be passed as selected item to the +options_for_select+ helper. # * The trhird parameter is used as prompt message in the select tag # * The last parameter is the method that is called on the page object to get the value that is passed with the params of the form. # def pages_for_select(pages = nil, selected = nil, prompt = "", page_attribute = :id) result = [[prompt.blank? ? t('Choose page') : prompt, ""]] if pages.blank? pages = Page.find_all_by_language_id_and_public(session[:language_id], true) end pages.each do |p| result << [p.name, p.send(page_attribute).to_s] end options_for_select(result, selected.to_s) end def render_essence_selection_editor(element, content, select_options) if content.class == String content = element.contents.find_by_name(content) else content = element.contents[content - 1] end if content.essence.nil? return warning('Element', t('content_essence_not_found')) end select_options = options_for_select(select_options, content.essence.content) select_tag( "contents[content_#{content.id}]", select_options, :class => 'alchemy_selectbox' ) end def admin_main_navigation entries = "" alchemy_modules.each do |alchemy_module| entries << alchemy_main_navigation_entry(alchemy_module) end entries.html_safe end def alchemy_main_navigation_entry(alchemy_module) render 'alchemy/admin/partials/main_navigation_entry', :alchemy_module => alchemy_module.stringify_keys, :navigation => alchemy_module['navigation'].stringify_keys end def admin_subnavigation alchemy_module = module_definition_for(:controller => params[:controller], :action => 'index') unless alchemy_module.nil? entries = alchemy_module["navigation"].stringify_keys['sub_navigation'] render_admin_subnavigation(entries) unless entries.nil? else "" end end # Renders the Subnavigation for the admin interface. def render_admin_subnavigation(entries) render "alchemy/admin/partials/sub_navigation_tab", :entries => entries end # Used for checking the main navi permissions def navigate_module(navigation) [navigation["action"].to_sym, navigation["controller"].gsub(/^\//, '').gsub(/\//, '_').to_sym] end # Returns true if the current controller and action is in a modules navigation definition. def admin_mainnavi_active?(mainnav) mainnav.stringify_keys! subnavi = mainnav["sub_navigation"].map(&:stringify_keys) if mainnav["sub_navigation"] nested = mainnav["nested"].map(&:stringify_keys) if mainnav["nested"] if subnavi (!subnavi.detect { |subnav| subnav["controller"].gsub(/^\//, '') == params[:controller] && subnav["action"] == params[:action] }.blank?) || (nested && !nested.detect { |n| n["controller"] == params[:controller] && n["action"] == params[:action] }.blank?) else mainnav["controller"] == params[:controller] && mainnav["action"] == params["action"] end end def admin_sub_navigation_entry_active?(entry) params[:controller] == entry["controller"].gsub(/^\//, '') && (params[:action] == entry["action"] || entry["nested_actions"] && entry["nested_actions"].include?(params[:action])) end # Calls the url_for helper on either an alchemy module engine, or the app alchemy is mounted at. def url_for_module(alchemy_module) navigation = alchemy_module['navigation'].stringify_keys url_options = { :controller => navigation['controller'], :action => navigation['action'] } if alchemy_module['engine_name'] eval(alchemy_module['engine_name']).url_for(url_options) else # hack to prefix any controller-path with / so it doesn't refer to alchemy/... url_options[:controller] = url_options[:controller].gsub(/^([^\/])/, "/#{$1}") main_app.url_for(url_options) end end # Calls the url_for helper on either an alchemy module engine, or the app alchemy is mounted at. def url_for_module_sub_navigation(navigation) alchemy_module = module_definition_for(navigation) engine_name = alchemy_module['engine_name'] if alchemy_module navigation.stringify_keys! url_options = { :controller => navigation['controller'], :action => navigation['action'] } if engine_name eval(engine_name).url_for(url_options) else main_app.url_for(url_options) end end def main_navigation_css_classes(navigation) ['main_navi_entry', admin_mainnavi_active?(navigation) ? 'active' : nil].compact.join(" ") end # (internal) Renders translated Module Names for html title element. def render_alchemy_title if content_for?(:title) title = content_for(:title) else title = t(controller_name, :scope => :libraries) end "Alchemy CMS - #{title}" end # (internal) Returns max image count as integer or nil. Used for the picture editor in element editor views. def max_image_count return nil if !@options if @options[:maximum_amount_of_images].blank? image_count = @options[:max_images] else image_count = @options[:maximum_amount_of_images] end if image_count.blank? nil else image_count.to_i end end # (internal) Renders a select tag for all items in the clipboard def clipboard_select_tag(items, html_options = {}) options = [[t('Please choose'), ""]] items.each do |item| options << [item.class.to_s == 'Alchemy::Element' ? item.display_name_with_preview_text : item.name, item.id] end select_tag( 'paste_from_clipboard', !@page.new_record? && @page.can_have_cells? ? grouped_elements_for_select(items, :id) : options_for_select(options), { :class => [html_options[:class], 'alchemy_selectbox'].join(' '), :style => html_options[:style] } ) end # Renders a toolbar button for the Alchemy toolbar # # == Options: # # :icon [String] # Icon class. See base.css.sccs for available icons, or make your own. # :label [String] # Text for button label. # :url [String] # Url for link. # :title [String] # Text for title tag. # :overlay [Boolean] # Pass true to open the link in a modal overlay window. # :overlay_options [Hash] # Overlay options. See link_to_overlay_window helper. # :if_permitted_to [Array] # Check permission for button. [:action, :controller]. Exactly how you defined the permission in your +authorization_rules.rb+. Defaults to controller and action from button url. # :skip_permission_check [Boolean] # Skip the permission check. Default false. NOT RECOMMENDED! # :loading_indicator [Boolean] # Shows the please wait overlay while loading. Default false. # def toolbar_button(options = {}) options.symbolize_keys! defaults = { :overlay => true, :skip_permission_check => false, :active => false, :link_options => {}, :overlay_options => {}, :loading_indicator => false } options = defaults.merge(options) button = content_tag('div', :class => 'button_with_label' + (options[:active] ? ' active' : '')) do link = if options[:overlay] link_to_overlay_window( render_icon(options[:icon]), options[:url], options[:overlay_options], { :class => 'icon_button', :title => options[:title] } ) else link_to options[:url], {:class => "icon_button#{options[:loading_indicator] ? nil : ' please_wait'}", :title => options[:title]}.merge(options[:link_options]) do render_icon(options[:icon]) end end link += content_tag('label', options[:label]) end if options[:skip_permission_check] return button else if options[:if_permitted_to].blank? action_controller = options[:url].gsub(/^\//, '').split('/') options[:if_permitted_to] = [action_controller.last.to_sym, action_controller[0..action_controller.length-2].join('_').to_sym] end if permitted_to?(*options[:if_permitted_to]) return button else return "" end end end # Renders the Alchemy backend toolbar # # == Options # # :buttons [Array] # Pass an Array with button options. They will be passed to toolbar_button helper. For options see toolbar_button # :search [Boolean] # Show searchfield. Default true. # def toolbar(options = {}) defaults = { :buttons => [], :search => true } options = defaults.merge(options) content_for(:toolbar) do content = <<-CONTENT #{options[:buttons].map { |button_options| toolbar_button(button_options) }.join()} #{render('alchemy/admin/partials/search_form', :url => options[:search_url]) if options[:search]} CONTENT content.html_safe end end # Renders the row for a resource record in the resources table. # # This helper has a nice fallback. If you create a partial for your record then this partial will be rendered. # # Otherwise the default +app/views/alchemy/admin/resources/_resource.html.erb+ partial gets rendered. # # == Example # # For a resource named +Comment+ you can create a partial named +_comment.html.erb+ # # # app/views/admin/comments/_comment.html.erb # # <%= comment.title %> # <%= comment.body %> # # # NOTE: Alchemy gives you a local variable named like your resource # def render_resources render :partial => resource_model_name, :collection => resources_instance_variable rescue ActionView::MissingTemplate render :partial => 'resource', :collection => resources_instance_variable end # (internal) Used by upload form def new_asset_path_with_session_information(asset_type) session_key = Rails.application.config.session_options[:key] if asset_type == "picture" alchemy.admin_pictures_path(session_key => cookies[session_key], request_forgery_protection_token => form_authenticity_token, :format => :js) elsif asset_type == "attachment" alchemy.admin_attachments_path(session_key => cookies[session_key], request_forgery_protection_token => form_authenticity_token, :format => :js) end end # Renders a textfield ready to display a datepicker # # Uses a HTML5 +input type="date"+ field. # # Pass a type as third option to override that. But old browsers hand this as text field anyway. So there is no need to override that. # # === Example # # <%= alchemy_datepicker(@person, :birthday) %> # def alchemy_datepicker(object, method, html_options={}) text_field(object.class.name.underscore.to_sym, method.to_sym, { :type => 'date', :class => 'thin_border date', :value => object.send(method.to_sym).nil? ? nil : l(object.send(method.to_sym), :format => :datepicker) }.merge(html_options)) end end end end