require 'active_admin/resource_collection' module ActiveAdmin # Namespaces are the basic organizing principle for resources within Active Admin # # Each resource is registered into a namespace which defines: # * the namespaceing for routing # * the module to namespace controllers # * the menu which gets displayed (other resources in the same namespace) # # For example: # # ActiveAdmin.register Post, namespace: :admin # # Will register the Post model into the "admin" namespace. This will namespace the # urls for the resource to "/admin/posts" and will set the controller to # Admin::PostsController # # You can also register to the "root" namespace, which is to say no namespace at all. # # ActiveAdmin.register Post, namespace: false # # This will register the resource to an instantiated namespace called :root. The # resource will be accessible from "/posts" and the controller will be PostsController. # class Namespace class << self def setting(name, default) Deprecation.warn "This method does not do anything and will be removed." end end RegisterEvent = 'active_admin.namespace.register'.freeze attr_reader :application, :resources, :menus def initialize(application, name) @application = application @name = name.to_s.underscore @resources = ResourceCollection.new register_module unless root? @menus = MenuCollection.new end def name @name.to_sym end def settings @settings ||= SettingsNode.build(application.namespace_settings) end def respond_to_missing?(method, include_private = false) settings.respond_to?(method) || super end def method_missing(method, *args) settings.respond_to?(method) ? settings.send(method, *args) : super end # Register a resource into this namespace. The preffered method to access this is to # use the global registration ActiveAdmin.register which delegates to the proper # namespace instance. def register(resource_class, options = {}, &block) config = find_or_build_resource(resource_class, options) # Register the resource register_resource_controller(config) parse_registration_block(config, &block) if block_given? # Dispatch a registration event ActiveSupport::Notifications.publish ActiveAdmin::Resource::RegisterEvent, config # Return the config config end # Add or update resource configuration options. # # Assumes corresponding Rails Controller class already exists. # Block is evaluated within config, not parsed as DSL. # # @param [Class] resource_class. ActiveRecord model. # # @return [Resource] def configure_resource(resource_class, options = {}, &block) config = find_or_build_resource(resource_class, options) raise "#{config.controller_name} not found" unless Object.const_defined?(config.controller_name) config.controller.active_admin_config = config yield(config) if block_given? ActiveSupport::Notifications.publish ActiveAdmin::Resource::RegisterEvent, config config end def register_page(name, options = {}, &block) config = build_page(name, options) # Register the resource register_page_controller(config) parse_page_registration_block(config, &block) if block_given? config end # Add page configuration options. # # Assumes corresponding Rails Controller class already exists. # Block is evaluated within config, not parsed as DSL. # # @param [String] name. # # @return [Resource] def configure_page(name, options = {}, &block) config = build_page(name, options) raise "#{config.controller_name} not found" unless Object.const_defined?(config.controller_name) config.controller.active_admin_config = config yield(config) if block_given? config end def root? name == :root end # Returns the name of the module if required. Will be nil if none # is required. # # eg: # Namespace.new(:admin).module_name # => 'Admin' # Namespace.new(:root).module_name # => nil # def module_name root? ? nil : @name.camelize end def route_prefix root? ? nil : @name end # Unload all the registered resources for this namespace def unload! @menus.clear! unload_resources! end # Returns the first registered ActiveAdmin::Resource instance for a given class def resource_for(klass) resources[klass] end def fetch_menu(name) build_menus! @menus.fetch(name) end # Add a callback to be ran when we build the menu # # @param [Symbol] name The name of the menu. Default: :default # @yield [ActiveAdmin::Menu] The block to be ran when the menu is built # # @return [void] def build_menu(name = DEFAULT_MENU, &block) extra_menus[name] = block end # The default logout menu item # # @param [ActiveAdmin::MenuItem] menu The menu to add the logout link to # @param [Fixnum] priority The numeric priority for the order in which it appears # @param [Hash] html_options An options hash to pass along to link_to # def add_logout_button_to_menu(menu, priority = 20, html_options = {}) if logout_link_path html_options = html_options.reverse_merge(method: logout_link_method || :get) menu.add id: 'logout', priority: priority, html_options: html_options, label: ->{ I18n.t 'active_admin.logout' }, url: ->{ render_or_call_method_or_proc_on self, active_admin_namespace.logout_link_path }, if: :current_active_admin_user? end end # The default user session menu item # # @param [ActiveAdmin::MenuItem] menu The menu to add the logout link to # @param [Fixnum] priority The numeric priority for the order in which it appears # @param [Hash] html_options An options hash to pass along to link_to # def add_current_user_to_menu(menu, priority = 10, html_options = {}) if current_user_method menu.add id: 'current_user', priority: priority, html_options: html_options, label: -> { display_name current_active_admin_user }, url: -> { auto_url_for(current_active_admin_user) }, if: :current_active_admin_user? end end protected def extra_menus @extra_menus ||= {} end def build_menus! # reset if any resources registered since last build @menus.clear! if resources.any? { |r| r.include_in_menu? && !r.menu_item } return if @menus.built? @menus.build_default_menu extra_menus.each_pair do |key, block| @menus.menu(key, &block) end build_default_utility_nav resources.each do |resource| resource.add_to_menu(@menus) end end # Builds the default utility navigation in top right header with current user & logout button def build_default_utility_nav return if @menus.exists? :utility_navigation @menus.menu :utility_navigation do |menu| add_current_user_to_menu menu add_logout_button_to_menu menu end end # Either returns an existing Resource instance or builds a new one. def find_or_build_resource(resource_class, options) resources.add Resource.new(self, resource_class, options) end def build_page(name, options) resources.add Page.new(self, name, options) end def register_page_controller(config) if Object.const_defined?("::#{config.controller_name}") unless config.controller.ancestors.include?(ActiveAdmin::PageController) raise "#{config.controller_name} not a PageController" end else eval "class ::#{config.controller_name} < ActiveAdmin::PageController; end" config.controller.const_set('DISPOSABLE', true) end config.controller.active_admin_config = config end def unload_resources! resources.each do |resource| parent = (module_name || 'Object').constantize name = resource.controller_name.split('::').last if parent.const_defined?(name, false) if resource.controller.const_defined?(:DISPOSABLE) parent.send(:remove_const, name) end end # Remove circular references resource.controller.active_admin_config = nil if resource.is_a?(Resource) && resource.dsl resource.dsl.run_registration_block { @config = nil } end end @resources = ResourceCollection.new end # Creates a ruby module to namespace all the classes in if required def register_module unless Object.const_defined? module_name Object.const_set module_name, Module.new end end def register_resource_controller(config) if Object.const_defined?("::#{config.controller_name}") unless config.controller.ancestors.include?(ActiveAdmin::ResourceController) raise "#{config.controller_name} not a ResourceController" end else eval "class ::#{config.controller_name} < ActiveAdmin::ResourceController; end" config.controller.const_set(:DISPOSABLE, true) end config.controller.active_admin_config = config end def parse_registration_block(config, &block) config.dsl = ResourceDSL.new(config) config.dsl.run_registration_block(&block) end def parse_page_registration_block(config, &block) PageDSL.new(config).run_registration_block(&block) end class Store include Enumerable delegate :[], :[]=, :empty?, to: :@namespaces def initialize @namespaces = {} end def each(&block) @namespaces.values.each(&block) end def names @namespaces.keys end end end end