Multiple Resources in an Extension
Multiple Resources in an Extension
This guide will show you how to:
- Create multiple tables on one single extension
WARNING: This only works on Refinery versions 3.0.0 and greater.
Generating an extension
Refinery ships with an extension generator that makes adding your own functionality a breeze. It works just like the Rails scaffold generator.
$ rails generate refinery:engine singular_model_name attribute:type [attribute:type ...]
TIP: to see all the options supported by the refinery:engine
generator just run rails g refinery:engine
.
Here is a list of the different field types are what they give you:
**field type** **description**
text a multiline visual editor
resource a link which pops open a dialog which allows the user to select an existing file or upload a new one
image a link which pops open a dialog which allows the user to select an existing image or upload a new one
string and integer a standard single line text input
If you remember, we told Rick that we'll give him an area to post up events he'll be at. Although he could technically create a new page in Refinery to add this content there, areas that have special functionality are much better suited as an extension.
Rick is going to want to enter the following information about each event:
- The event title
- The event date
- A photo
- A little blurb about the event.
Run this command to generate the events extension for Rick:
$ rails generate refinery:engine event title:string date:datetime photo:image blurb:text
TIP: if you want to create a model without any front-end code (i.e. code only for the administrative interface), add --skip-frontend
.
TIP: if you want to use a custom namespace for your extension, add --namespace <namespace_name>
. If you use it, you have to remember to specify this namespace for every scaffold you generate for this extension.
This results in the following:
create vendor/extensions/events/tasks/rspec.rake
create vendor/extensions/events/tasks/testing.rake
create vendor/extensions/events/app/models/refinery/events/event.rb
create vendor/extensions/events/app/controllers/refinery/events/events_controller.rb
create vendor/extensions/events/app/controllers/refinery/events/admin/events_controller.rb
create vendor/extensions/events/app/views/refinery/events/admin/events/index.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/edit.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/_event.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/_records.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/_form.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/new.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/_events.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/_sortable_list.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/events/_actions.html.erb
create vendor/extensions/events/app/views/refinery/events/events/index.html.erb
create vendor/extensions/events/app/views/refinery/events/events/show.html.erb
create vendor/extensions/events/config/locales/cs.yml
create vendor/extensions/events/config/locales/nb.yml
create vendor/extensions/events/config/locales/es.yml
create vendor/extensions/events/config/locales/sk.yml
create vendor/extensions/events/config/locales/it.yml
create vendor/extensions/events/config/locales/en.yml
create vendor/extensions/events/config/locales/ru.yml
create vendor/extensions/events/config/locales/fr.yml
create vendor/extensions/events/config/locales/tr.yml
create vendor/extensions/events/config/locales/zh-CN.yml
create vendor/extensions/events/config/locales/nl.yml
create vendor/extensions/events/config/routes.rb
create vendor/extensions/events/spec/spec_helper.rb
create vendor/extensions/events/spec/features/refinery/events/admin/events_spec.rb
create vendor/extensions/events/spec/models/refinery/events/event_spec.rb
create vendor/extensions/events/spec/support/factories/refinery/events.rb
create vendor/extensions/events/refinerycms-events.gemspec
create vendor/extensions/events/script/rails
create vendor/extensions/events/readme.md
create vendor/extensions/events/Rakefile
create vendor/extensions/events/lib/tasks/refinery/events.rake
create vendor/extensions/events/lib/refinerycms-events.rb
create vendor/extensions/events/lib/refinery/events.rb
create vendor/extensions/events/lib/refinery/events/engine.rb
create vendor/extensions/events/lib/generators/refinery/events_generator.rb
create vendor/extensions/events/db/migrate/1_create_events_events.rb
create vendor/extensions/events/Gemfile
append /Users/rick/Sites/refinery/refinerycms/Gemfile
------------------------
Now run:
bundle install
rails generate refinery:events
rake db:migrate
rake db:seed
Please restart your rails server.
------------------------
As the output shows, next run:
$ bundle install
$ rails generate refinery:events
$ rake db:migrate
$ rake db:seed
A refinery:events
generator is created for you to install the migration to create the events table. Run all the commands provided in the terminal.
TIP: When new extensions are added, it's a good idea to restart your server for new changes to be loaded in.
As you can see inside your text editor, the event extension is now stored in vendor/extensions/events/
. This is where all your files – including your migration file – will be placed. The folder structure of events
is nearly identical to a normal Rails app, but has a few additions to provide Refinery functionality.
Once you have generated an extension, it's time to create another scaffold to put inside of it.
To do that, run the following command.
$ rails g refinery:engine place name:string --extension events --namespace events
TIP: You can additionally specify --pretend
to simulate generation, so you may inspect the outcome without actually modifying anything.
Notice the last arguments (--extension <extension_name> --namespace <extension_name>
). This is how Refinery knows which extension to insert your new code. The --namespace
argument is necessary because Refinery will create a namespace for your extension by default. If you don't specify one, it's the name of the first scaffold you created. In this
case, the namespace is events
. If you look inside, for example, vendor/extensions/events/app/controllers/refinery/events/events_controller.rb
, you will see that the opening lines look something like this:
module Refinery
module Events
class EventsController < ::ApplicationController
The first two lines indicate that the extension is namespaced using Refinery::Events
. Refinery will automatically add the Refinery
for you, but you will have to manually specify the namespace.
Running this command will produce the following output:
identical vendor/extensions/events/tasks/rspec.rake
identical vendor/extensions/events/tasks/testing.rake
create vendor/extensions/events/app/models/refinery/events/place.rb
create vendor/extensions/events/app/controllers/refinery/events/places_controller.rb
create vendor/extensions/events/app/controllers/refinery/events/admin/places_controller.rb
create vendor/extensions/events/app/views/refinery/events/admin/places/index.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/edit.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/_place.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/_records.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/_form.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/new.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/_places.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/_sortable_list.html.erb
create vendor/extensions/events/app/views/refinery/events/admin/places/_actions.html.erb
create vendor/extensions/events/app/views/refinery/events/places/index.html.erb
create vendor/extensions/events/app/views/refinery/events/places/show.html.erb
create vendor/extensions/events/config/locales/tmp/cs.yml
create vendor/extensions/events/config/locales/tmp/nb.yml
create vendor/extensions/events/config/locales/tmp/es.yml
create vendor/extensions/events/config/locales/tmp/sk.yml
create vendor/extensions/events/config/locales/tmp/it.yml
create vendor/extensions/events/config/locales/tmp/en.yml
create vendor/extensions/events/config/locales/tmp/ru.yml
create vendor/extensions/events/config/locales/tmp/fr.yml
create vendor/extensions/events/config/locales/tmp/tr.yml
create vendor/extensions/events/config/locales/tmp/zh-CN.yml
create vendor/extensions/events/config/locales/tmp/nl.yml
create vendor/extensions/events/config/tmp/routes.rb
identical vendor/extensions/events/spec/spec_helper.rb
create vendor/extensions/events/spec/features/refinery/events/admin/places_spec.rb
create vendor/extensions/events/spec/models/refinery/events/place_spec.rb
create vendor/extensions/events/spec/support/factories/refinery/places.rb
identical vendor/extensions/events/refinerycms-events.gemspec
identical vendor/extensions/events/script/rails
identical vendor/extensions/events/readme.md
identical vendor/extensions/events/Rakefile
identical vendor/extensions/events/lib/tasks/refinery/events.rake
create vendor/extensions/events/lib/tmp/refinerycms-events.rb
create vendor/extensions/events/lib/refinery/places.rb
create vendor/extensions/events/lib/refinery/places/engine.rb
identical vendor/extensions/events/lib/generators/refinery/events_generator.rb
create vendor/extensions/events/db/migrate/2_create_events_places.rb
identical vendor/extensions/events/Gemfile
------------------------
Now run:
bundle install
rails generate refinery:events
rake db:migrate
rake db:seed
Please restart your rails server.
------------------------
WARNING: If you are presented with a conflict in the events_generator.rb
file, say no! This happens at the moment because Refinery thinks you are generating a Places extension, and this may cause all kinds of havoc if you agree to it. If you have accidentally agreed to it, you can revert that file, and check your db/seeds.rb
file to see if you have accidentally appended an additional line reading Refinery::Events::Engine.load_seed
.
Run the commands listed above. Notice rails generate refinery:events
. The rails generate refinery:events
will copy the migration files from vendor/extensions/events/db/migrate
to db/migrate
and will prepend a timestamp to each migration. This is so an extension that was written before your application will declare its migrations chronologically
later than the migrations you wrote for the app. For instance, if you wrote your app on Tuesday and then inserted the extension on Wednesday, you would want the timestamps on the migrations to show Wednesday rather than the date it was created – perhaps Monday. If the extension didn't do this, when you deployed your application, it would run the migrations for your extension first, potentially failing because it couldn't find the Refinery database tables it needs to complete.
Crudify: The Backbone of Refinery Engines
Any Refinery extension, even the built-in ones, that focuses on the standard create, read, update and delete operations are driven by crudify
. Crudify is a highly reusable module included with Refinery that gives you all the standard CRUD actions as well as reordering, searching and paging.
Open up vendor/extensions/events/app/controllers/refinery/events/admin/events_controller.rb
and look at its contents:
module Refinery
module Events
module Admin
class EventsController < ::Refinery::AdminController
crudify :'refinery/events/event'
end
end
end
end
Most of the time, crudify's defaults are bang on, but if you need to, you can easily customise how it works.
By default crudify
assumes your records will be sortable. But events should not be manually sortable; it makes more sense to order them by their event date. Update the contents of the file to this:
module Refinery
module Events
module Admin
class EventsController < ::Refinery::AdminController
crudify :'refinery/events/event', order: "date DESC", sortable: false
end
end
end
end
This will tell crudify
to sort by our event date field and to turn off manual sorting by the user.
Finally edit vendor/extensions/events/app/controllers/refinery/events/events_controller.rb
and replace the find_all_events
method with this one:
module Refinery
module Events
class EventsController < ::ApplicationController
# code
protected
def find_all_events
# Order by event date
@events = Event.order("date DESC")
end
# code
end
end
end
This will sort your events by date, rather than the default, which is its ID (a rather information-less order).