lib/active_admin/resource/routes.rb



# frozen_string_literal: true
module ActiveAdmin
  class Resource
    module Routes
      # @param params [Hash] of params: { study_id: 3 }
      # @return [String] the path to this resource collection page
      # @example "/admin/posts"
      def route_collection_path(params = {}, additional_params = {})
        route_builder.collection_path(params, additional_params)
      end

      def route_batch_action_path(params = {}, additional_params = {})
        route_builder.batch_action_path(params, additional_params)
      end

      # @param resource [ActiveRecord::Base] the instance we want the path of
      # @return [String] the path to this resource collection page
      # @example "/admin/posts/1"
      def route_instance_path(resource, additional_params = {})
        route_builder.instance_path(resource, additional_params)
      end

      def route_edit_instance_path(resource, additional_params = {})
        route_builder.member_action_path(:edit, resource, additional_params)
      end

      def route_member_action_path(action, resource, additional_params = {})
        route_builder.member_action_path(action, resource, additional_params)
      end

      # Returns the routes prefix for this config
      def route_prefix
        namespace.route_prefix
      end

      def route_builder
        @route_builder ||= RouteBuilder.new(self)
      end

      def route_uncountable?
        config = resources_configuration[:self]

        config[:route_collection_name] == config[:route_instance_name]
      end

      class RouteBuilder
        def initialize(resource)
          @resource = resource
        end

        def collection_path(params, additional_params = {})
          route_name = route_name(
            resource.resources_configuration[:self][:route_collection_name],
            suffix: (resource.route_uncountable? ? "index_path" : "path")
          )

          routes.public_send route_name, *route_collection_params(params), additional_params
        end

        def batch_action_path(params, additional_params = {})
          route_name = route_name(
            resource.resources_configuration[:self][:route_collection_name],
            action: :batch_action,
            suffix: (resource.route_uncountable? ? "index_path" : "path")
          )

          query = params.slice(:q, :scope)
          query = query.permit!.to_h
          routes.public_send route_name, *route_collection_params(params), additional_params.merge(query)
        end

        # @return [String] the path to this resource collection page
        # @param instance [ActiveRecord::Base] the instance we want the path of
        # @example "/admin/posts/1"
        def instance_path(instance, additional_params = {})
          route_name = route_name(resource.resources_configuration[:self][:route_instance_name])

          routes.public_send route_name, *route_instance_params(instance), additional_params
        end

        # @return [String] the path to the member action of this resource
        # @param action [Symbol]
        # @param instance [ActiveRecord::Base] the instance we want the path of
        # @example "/admin/posts/1/edit"
        def member_action_path(action, instance, additional_params = {})
          path = resource.resources_configuration[:self][:route_instance_name]
          route_name = route_name(path, action: action)

          routes.public_send route_name, *route_instance_params(instance), additional_params
        end

        private

        attr_reader :resource

        def route_name(resource_path_name, options = {})
          suffix = options[:suffix] || "path"
          route = []

          route << options[:action] # "batch_action", "edit" or "new"
          route << resource.route_prefix # "admin"
          route << belongs_to_name if nested? # "category"
          route << resource_path_name # "posts" or "post"
          route << suffix # "path" or "index path"

          route.compact.join("_").to_sym # :admin_category_posts_path
        end

        # @return params to pass to instance path
        def route_instance_params(instance)
          if nested?
            [instance.public_send(belongs_to_target_name).to_param, instance.to_param]
          else
            instance.to_param
          end
        end

        def route_collection_params(params)
          if nested?
            params[:"#{belongs_to_name}_id"]
          end
        end

        def nested?
          resource.belongs_to? && belongs_to_config.required?
        end

        def belongs_to_target_name
          belongs_to_config.target_name
        end

        def belongs_to_name
          belongs_to_config.target.resource_name.singular
        end

        def belongs_to_config
          resource.belongs_to_config
        end

        def routes
          Helpers::Routes
        end
      end
    end
  end
end