layout: default nav_order: 8 title: Custom Actions

redirect_from: /docs/8-custom-actions.html

Custom Controller Actions

Add custom actions the Rails Way:

Collection Actions

A collection action is a controller action which operates on the collection of resources. Add both the action to the controller as well as a route.

class Admin::PostsController < ActiveAdmin::ResourceController
  def import_csv
    # Do some CSV importing work here...
    redirect_to collection_path, notice: "CSV imported successfully!"
  end
end

and

Rails.application.routes.draw do
  post '/admin/posts/import_csv', controller: 'admin/posts', action: 'import_csv'
  ...

This collection action will generate a route at /admin/posts/import_csv pointing to the Admin::PostsController#import_csv controller action.

Member Actions

A member action is a controller action which operates on a single resource.

For example, to add a lock action to a user resource, you would do the following:

class Admin::UsersController < ActiveAdmin::ResourceController
  def lock
    resource.lock!
    redirect_to resource_path, notice: "Locked!"
  end

and

Rails.application.routes.draw do
  put '/admin/users/:id/lock', controller: 'admin/users', action: 'lock'
  ...

HTTP Verbs

Sometimes you want to create an action that handles multiple HTTP verbs. In that case, this is the suggested approach:

def foo
  if request.post?
    resource.update_attributes! foo: params[:foo] || {}
    head :ok
  else
    render :foo
  end
end

and

Rails.application.routes.draw do
  get 'admin/users/:id/foo', container: 'admin/users', action: 'foo'
  post 'admin/users/:id/foo', container: 'admin/users', action: 'foo'
  ...

Rendering

Custom controller actions support rendering within the standard Active Admin layout.

class Admin::PostsController < ActiveAdmin::ResourceController
  # /admin/posts/:id/comments
  def comments
    @comments = resource.comments
    # This will render app/views/admin/posts/comments.html.erb
  end
end

If you would like to use the same view syntax as the rest of Active Admin, you can use the Arbo file extension: .arb.

For example, create app/views/admin/posts/comments.html.arb with:

table_for assigns[:post].comments do
  column :id
  column :author
  column :body do |comment|
    simple_format comment.body
  end
end

Page Titles

The page title for the custom action will be the translated version of the controller action name. For example, a member_action named "upload_csv" will look up a translation key of active_admin.upload_csv. If none are found, it defaults to the name of the controller action.

If this doesn't work for you, you can always set the @page_title instance variable in your controller action to customize the page title.

class Admin::PostsController < ActiveAdmin::ResourceController
  def comments
    @comments   = resource.comments
    @page_title = "#{resource.title}: Comments" # Sets the page title
  end
end

Alternately use content_for(:page_title) in the view template.

Action Items

By default action items (like the New, Edit and Delete buttons) are rendered in a partial included in the right hand side of the title bar.

For example, to add a "View on site" button to view a blog post customize the action_item partial as follows:

# app/views/admin/posts/_action_item.html.arb
div(class: :action_items) do
  if params[:action] == 'index'
    if controller.action_methods.include?('new') && authorized?(:create, Post)
      action_link :new_model, new_resource_path
    end
  end

  if params[:action] == 'show'
    if controller.action_methods.include?('edit') && authorized?(:update, resource)
      action_link :edit_model, edit_resource_path(resource)
    end

    if resource.published? && current_admin_user.super_admin?
      action_link "View on site", post_path(resource)
    end
  end
end

Modifying the Controller

Using the controller method within the registration block is deprecated, create and modify the controller explicitly.