pager-resource_controller 1.0.20080513
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/LICENSE +22 -0
- data/README +276 -0
- data/Rakefile +55 -0
- data/TODO +1 -0
- data/generators/scaffold_resource/USAGE +29 -0
- data/generators/scaffold_resource/scaffold_resource_generator.rb +101 -0
- data/generators/scaffold_resource/templates/controller.rb +2 -0
- data/generators/scaffold_resource/templates/fixtures.yml +10 -0
- data/generators/scaffold_resource/templates/functional_test.rb +57 -0
- data/generators/scaffold_resource/templates/helper.rb +2 -0
- data/generators/scaffold_resource/templates/migration.rb +15 -0
- data/generators/scaffold_resource/templates/model.rb +2 -0
- data/generators/scaffold_resource/templates/old_migration.rb +13 -0
- data/generators/scaffold_resource/templates/shoulda_functional_test.rb +19 -0
- data/generators/scaffold_resource/templates/unit_test.rb +7 -0
- data/generators/scaffold_resource/templates/view__form.erb +6 -0
- data/generators/scaffold_resource/templates/view__form.haml +5 -0
- data/generators/scaffold_resource/templates/view_edit.erb +16 -0
- data/generators/scaffold_resource/templates/view_edit.haml +11 -0
- data/generators/scaffold_resource/templates/view_index.erb +22 -0
- data/generators/scaffold_resource/templates/view_index.haml +19 -0
- data/generators/scaffold_resource/templates/view_new.erb +12 -0
- data/generators/scaffold_resource/templates/view_new.haml +9 -0
- data/generators/scaffold_resource/templates/view_show.erb +9 -0
- data/generators/scaffold_resource/templates/view_show.haml +9 -0
- data/init.rb +6 -0
- data/install.rb +1 -0
- data/lib/resource_controller.rb +11 -0
- data/lib/resource_controller/accessors.rb +76 -0
- data/lib/resource_controller/action_options.rb +28 -0
- data/lib/resource_controller/actions.rb +75 -0
- data/lib/resource_controller/base.rb +15 -0
- data/lib/resource_controller/class_methods.rb +22 -0
- data/lib/resource_controller/controller.rb +63 -0
- data/lib/resource_controller/failable_action_options.rb +17 -0
- data/lib/resource_controller/helpers.rb +28 -0
- data/lib/resource_controller/helpers/current_objects.rb +69 -0
- data/lib/resource_controller/helpers/internal.rb +59 -0
- data/lib/resource_controller/helpers/nested.rb +45 -0
- data/lib/resource_controller/helpers/urls.rb +124 -0
- data/lib/resource_controller/response_collector.rb +21 -0
- data/lib/urligence.rb +50 -0
- data/rails/init.rb +1 -0
- data/test/Rakefile +10 -0
- data/test/app/controllers/application.rb +7 -0
- data/test/app/controllers/cms/options_controller.rb +3 -0
- data/test/app/controllers/cms/products_controller.rb +2 -0
- data/test/app/controllers/comments_controller.rb +3 -0
- data/test/app/controllers/people_controller.rb +9 -0
- data/test/app/controllers/photos_controller.rb +10 -0
- data/test/app/controllers/posts_controller.rb +10 -0
- data/test/app/controllers/projects_controller.rb +3 -0
- data/test/app/controllers/somethings_controller.rb +3 -0
- data/test/app/controllers/tags_controller.rb +13 -0
- data/test/app/controllers/users_controller.rb +12 -0
- data/test/app/helpers/application_helper.rb +3 -0
- data/test/app/helpers/cms/products_helper.rb +2 -0
- data/test/app/helpers/comments_helper.rb +2 -0
- data/test/app/helpers/options_helper.rb +2 -0
- data/test/app/helpers/people_helper.rb +2 -0
- data/test/app/helpers/photos_helper.rb +2 -0
- data/test/app/helpers/posts_helper.rb +2 -0
- data/test/app/helpers/projects_helper.rb +2 -0
- data/test/app/helpers/somethings_helper.rb +2 -0
- data/test/app/helpers/tags_helper.rb +2 -0
- data/test/app/helpers/users_helper.rb +2 -0
- data/test/app/models/account.rb +3 -0
- data/test/app/models/comment.rb +3 -0
- data/test/app/models/option.rb +3 -0
- data/test/app/models/photo.rb +4 -0
- data/test/app/models/post.rb +3 -0
- data/test/app/models/product.rb +3 -0
- data/test/app/models/project.rb +2 -0
- data/test/app/models/something.rb +2 -0
- data/test/app/models/tag.rb +3 -0
- data/test/app/views/cms/options/edit.rhtml +17 -0
- data/test/app/views/cms/options/index.rhtml +20 -0
- data/test/app/views/cms/options/new.rhtml +16 -0
- data/test/app/views/cms/options/show.rhtml +8 -0
- data/test/app/views/cms/products/edit.rhtml +17 -0
- data/test/app/views/cms/products/index.rhtml +20 -0
- data/test/app/views/cms/products/new.rhtml +16 -0
- data/test/app/views/cms/products/show.rhtml +8 -0
- data/test/app/views/comments/edit.rhtml +27 -0
- data/test/app/views/comments/index.rhtml +24 -0
- data/test/app/views/comments/new.rhtml +26 -0
- data/test/app/views/comments/show.rhtml +18 -0
- data/test/app/views/layouts/application.rhtml +17 -0
- data/test/app/views/layouts/comments.rhtml +17 -0
- data/test/app/views/layouts/options.rhtml +17 -0
- data/test/app/views/layouts/people.rhtml +17 -0
- data/test/app/views/layouts/photos.rhtml +17 -0
- data/test/app/views/layouts/projects.rhtml +17 -0
- data/test/app/views/layouts/somethings.rhtml +17 -0
- data/test/app/views/layouts/tags.rhtml +17 -0
- data/test/app/views/people/edit.rhtml +17 -0
- data/test/app/views/people/index.rhtml +20 -0
- data/test/app/views/people/new.rhtml +16 -0
- data/test/app/views/people/show.rhtml +8 -0
- data/test/app/views/photos/edit.rhtml +17 -0
- data/test/app/views/photos/index.rhtml +20 -0
- data/test/app/views/photos/new.rhtml +16 -0
- data/test/app/views/photos/show.rhtml +8 -0
- data/test/app/views/posts/edit.rhtml +22 -0
- data/test/app/views/posts/index.rhtml +22 -0
- data/test/app/views/posts/new.rhtml +21 -0
- data/test/app/views/posts/show.rhtml +13 -0
- data/test/app/views/projects/edit.rhtml +17 -0
- data/test/app/views/projects/index.rhtml +20 -0
- data/test/app/views/projects/new.rhtml +16 -0
- data/test/app/views/projects/show.rhtml +8 -0
- data/test/app/views/somethings/edit.rhtml +17 -0
- data/test/app/views/somethings/index.rhtml +20 -0
- data/test/app/views/somethings/new.rhtml +16 -0
- data/test/app/views/somethings/show.rhtml +8 -0
- data/test/app/views/tags/edit.rhtml +17 -0
- data/test/app/views/tags/index.rhtml +20 -0
- data/test/app/views/tags/index.rjs +0 -0
- data/test/app/views/tags/new.rhtml +16 -0
- data/test/app/views/tags/show.rhtml +8 -0
- data/test/app/views/users/edit.rhtml +17 -0
- data/test/app/views/users/index.rhtml +20 -0
- data/test/app/views/users/new.rhtml +16 -0
- data/test/app/views/users/show.rhtml +8 -0
- data/test/config/boot.rb +45 -0
- data/test/config/database.yml +16 -0
- data/test/config/environment.rb +64 -0
- data/test/config/environments/development.rb +21 -0
- data/test/config/environments/test.rb +19 -0
- data/test/config/routes.rb +51 -0
- data/test/db/migrate/001_create_posts.rb +12 -0
- data/test/db/migrate/002_create_products.rb +11 -0
- data/test/db/migrate/003_create_comments.rb +13 -0
- data/test/db/migrate/004_create_options.rb +12 -0
- data/test/db/migrate/005_create_photos.rb +11 -0
- data/test/db/migrate/006_create_tags.rb +17 -0
- data/test/db/migrate/007_create_somethings.rb +11 -0
- data/test/db/migrate/008_create_accounts.rb +11 -0
- data/test/db/migrate/009_add_account_id_to_photos.rb +9 -0
- data/test/db/migrate/010_create_projects.rb +11 -0
- data/test/db/schema.rb +65 -0
- data/test/script/console +3 -0
- data/test/script/destroy +3 -0
- data/test/script/generate +3 -0
- data/test/script/server +3 -0
- data/test/test/fixtures/accounts.yml +7 -0
- data/test/test/fixtures/comments.yml +11 -0
- data/test/test/fixtures/options.yml +9 -0
- data/test/test/fixtures/photos.yml +9 -0
- data/test/test/fixtures/photos_tags.yml +3 -0
- data/test/test/fixtures/posts.yml +9 -0
- data/test/test/fixtures/products.yml +7 -0
- data/test/test/fixtures/projects.yml +7 -0
- data/test/test/fixtures/somethings.yml +7 -0
- data/test/test/fixtures/tags.yml +7 -0
- data/test/test/functional/cms/options_controller_test.rb +20 -0
- data/test/test/functional/cms/products_controller_test.rb +18 -0
- data/test/test/functional/comments_controller_test.rb +26 -0
- data/test/test/functional/people_controller_test.rb +34 -0
- data/test/test/functional/photos_controller_test.rb +128 -0
- data/test/test/functional/posts_controller_test.rb +34 -0
- data/test/test/functional/projects_controller_test.rb +18 -0
- data/test/test/functional/somethings_controller_test.rb +28 -0
- data/test/test/functional/tags_controller_test.rb +64 -0
- data/test/test/functional/users_controller_test.rb +24 -0
- data/test/test/test_helper.rb +12 -0
- data/test/test/unit/accessors_test.rb +91 -0
- data/test/test/unit/account_test.rb +7 -0
- data/test/test/unit/action_options_test.rb +66 -0
- data/test/test/unit/base_test.rb +11 -0
- data/test/test/unit/comment_test.rb +10 -0
- data/test/test/unit/failable_action_options_test.rb +50 -0
- data/test/test/unit/helpers/current_objects_test.rb +127 -0
- data/test/test/unit/helpers/internal_test.rb +88 -0
- data/test/test/unit/helpers/nested_test.rb +82 -0
- data/test/test/unit/helpers/urls_test.rb +71 -0
- data/test/test/unit/helpers_test.rb +25 -0
- data/test/test/unit/option_test.rb +10 -0
- data/test/test/unit/photo_test.rb +10 -0
- data/test/test/unit/post_test.rb +10 -0
- data/test/test/unit/project_test.rb +10 -0
- data/test/test/unit/response_collector_test.rb +31 -0
- data/test/test/unit/something_test.rb +10 -0
- data/test/test/unit/tag_test.rb +10 -0
- data/test/test/unit/urligence_test.rb +203 -0
- data/test/vendor/plugins/shoulda/README +123 -0
- data/test/vendor/plugins/shoulda/Rakefile +29 -0
- data/test/vendor/plugins/shoulda/bin/convert_to_should_syntax +40 -0
- data/test/vendor/plugins/shoulda/init.rb +3 -0
- data/test/vendor/plugins/shoulda/lib/shoulda.rb +47 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/active_record_helpers.rb +338 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/color.rb +77 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/context.rb +143 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/controller_tests/controller_tests.rb +470 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/controller_tests/formats/html.rb +192 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/controller_tests/formats/xml.rb +162 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/general.rb +119 -0
- data/test/vendor/plugins/shoulda/lib/shoulda/private_helpers.rb +17 -0
- data/test/vendor/plugins/shoulda/tasks/list_tests.rake +40 -0
- data/uninstall.rb +1 -0
- metadata +410 -0
@@ -0,0 +1,15 @@
|
|
1
|
+
module ResourceController
|
2
|
+
|
3
|
+
# == ResourceController::Base
|
4
|
+
#
|
5
|
+
# Inherit from this class to create your RESTful controller. See the README for usage.
|
6
|
+
#
|
7
|
+
class Base < ApplicationController
|
8
|
+
unloadable
|
9
|
+
|
10
|
+
def self.inherited(subclass)
|
11
|
+
super
|
12
|
+
subclass.class_eval { resource_controller }
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
module ResourceController
|
2
|
+
module ClassMethods
|
3
|
+
|
4
|
+
# Use this method in your controller to specify which actions you'd like it to respond to.
|
5
|
+
#
|
6
|
+
# class PostsController < ResourceController::Base
|
7
|
+
# actions :all, :except => :create
|
8
|
+
# end
|
9
|
+
def actions(*opts)
|
10
|
+
config = {}
|
11
|
+
config.merge!(opts.pop) if opts.last.is_a?(Hash)
|
12
|
+
|
13
|
+
actions_to_remove = []
|
14
|
+
actions_to_remove += (ResourceController::ACTIONS - [:new_action] + [:new]) - opts unless opts.first == :all
|
15
|
+
actions_to_remove += [*config[:except]] if config[:except]
|
16
|
+
actions_to_remove.uniq!
|
17
|
+
|
18
|
+
actions_to_remove.each { |action| undef_method(action)}
|
19
|
+
end
|
20
|
+
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,63 @@
|
|
1
|
+
module ResourceController
|
2
|
+
module Controller
|
3
|
+
def self.included(subclass)
|
4
|
+
subclass.class_eval do
|
5
|
+
include ResourceController::Helpers
|
6
|
+
include ResourceController::Actions
|
7
|
+
extend ResourceController::Accessors
|
8
|
+
extend ResourceController::ClassMethods
|
9
|
+
|
10
|
+
class_reader_writer :belongs_to, *NAME_ACCESSORS
|
11
|
+
NAME_ACCESSORS.each { |accessor| send(accessor, controller_name.singularize.underscore) }
|
12
|
+
|
13
|
+
ACTIONS.each do |action|
|
14
|
+
class_scoping_reader action, FAILABLE_ACTIONS.include?(action) ? FailableActionOptions.new : ActionOptions.new
|
15
|
+
end
|
16
|
+
|
17
|
+
self.helper_method :object_url, :edit_object_url, :new_object_url, :collection_url, :object, :collection,
|
18
|
+
:parent, :parent_type, :parent_object, :model_name, :model, :object_path, :edit_object_path,
|
19
|
+
:new_object_path, :collection_path, :hash_for_collection_path, :hash_for_object_path,
|
20
|
+
:hash_for_edit_object_path, :hash_for_new_object_path, :hash_for_collection_url,
|
21
|
+
:hash_for_object_url, :hash_for_edit_object_url, :hash_for_new_object_url, :parent?,
|
22
|
+
:collection_url_options, :object_url_options, :new_object_url_options
|
23
|
+
|
24
|
+
end
|
25
|
+
|
26
|
+
init_default_actions(subclass)
|
27
|
+
end
|
28
|
+
|
29
|
+
private
|
30
|
+
def self.init_default_actions(klass)
|
31
|
+
klass.class_eval do
|
32
|
+
index.wants.html
|
33
|
+
edit.wants.html
|
34
|
+
new_action.wants.html
|
35
|
+
|
36
|
+
show do
|
37
|
+
wants.html
|
38
|
+
|
39
|
+
failure.wants.html { render :text => "Member object not found." }
|
40
|
+
end
|
41
|
+
|
42
|
+
create do
|
43
|
+
flash "Successfully created!"
|
44
|
+
wants.html { redirect_to object_url }
|
45
|
+
|
46
|
+
failure.wants.html { render :action => "new" }
|
47
|
+
end
|
48
|
+
|
49
|
+
update do
|
50
|
+
flash "Successfully updated!"
|
51
|
+
wants.html { redirect_to object_url }
|
52
|
+
|
53
|
+
failure.wants.html { render :action => "edit" }
|
54
|
+
end
|
55
|
+
|
56
|
+
destroy do
|
57
|
+
flash "Successfully removed!"
|
58
|
+
wants.html { redirect_to collection_url }
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
@@ -0,0 +1,17 @@
|
|
1
|
+
module ResourceController
|
2
|
+
class FailableActionOptions
|
3
|
+
extend ResourceController::Accessors
|
4
|
+
|
5
|
+
scoping_reader :success, :fails
|
6
|
+
alias_method :failure, :fails
|
7
|
+
|
8
|
+
block_accessor :before
|
9
|
+
|
10
|
+
def initialize
|
11
|
+
@success = ActionOptions.new
|
12
|
+
@fails = ActionOptions.new
|
13
|
+
end
|
14
|
+
|
15
|
+
delegate :flash, :after, :response, :wants, :to => :success
|
16
|
+
end
|
17
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
module ResourceController
|
2
|
+
# == ResourceController::Helpers
|
3
|
+
#
|
4
|
+
# Included in Base.
|
5
|
+
#
|
6
|
+
# These helpers are used internally to manage objects, generate urls, and manage parent resource associations.
|
7
|
+
#
|
8
|
+
# If you want to customize certain controller behaviour, like member-object, and collection fetching, overriding these methods is all it takes.
|
9
|
+
#
|
10
|
+
# See the docs below, and the README for examples
|
11
|
+
#
|
12
|
+
# *Please Note: many of these helpers build on top of each other, and require that behaviour to be maintained, in order for other functionality to work properly.*
|
13
|
+
#
|
14
|
+
# e.g. All fetching must be done on top of the method end_of_association_chain, or else parent resources (including polymorphic ones) won't function correctly.
|
15
|
+
#
|
16
|
+
# class PostsController < ResourceController::Base
|
17
|
+
# private
|
18
|
+
# def object
|
19
|
+
# @object ||= end_of_association_chain.find_by_permalink(param)
|
20
|
+
# end
|
21
|
+
# end
|
22
|
+
module Helpers
|
23
|
+
include ResourceController::Helpers::Urls
|
24
|
+
include ResourceController::Helpers::Internal
|
25
|
+
include ResourceController::Helpers::Nested
|
26
|
+
include ResourceController::Helpers::CurrentObjects
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,69 @@
|
|
1
|
+
module ResourceController::Helpers::CurrentObjects
|
2
|
+
protected
|
3
|
+
# Used internally to return the model for your resource.
|
4
|
+
#
|
5
|
+
def model
|
6
|
+
model_name.to_s.camelize.constantize
|
7
|
+
end
|
8
|
+
|
9
|
+
|
10
|
+
# Used to fetch the collection for the index method
|
11
|
+
#
|
12
|
+
# In order to customize the way the collection is fetched, to add something like pagination, for example, override this method.
|
13
|
+
#
|
14
|
+
def collection
|
15
|
+
end_of_association_chain.find(:all)
|
16
|
+
end
|
17
|
+
|
18
|
+
# Returns the current param.
|
19
|
+
#
|
20
|
+
# Defaults to params[:id].
|
21
|
+
#
|
22
|
+
# Override this method if you'd like to use an alternate param name.
|
23
|
+
#
|
24
|
+
def param
|
25
|
+
params[:id]
|
26
|
+
end
|
27
|
+
|
28
|
+
# Used to fetch the current member object in all of the singular methods that operate on an existing member.
|
29
|
+
#
|
30
|
+
# Override this method if you'd like to fetch your objects in some alternate way, like using a permalink.
|
31
|
+
#
|
32
|
+
# class PostsController < ResourceController::Base
|
33
|
+
# private
|
34
|
+
# def object
|
35
|
+
# @object ||= end_of_association_chain.find_by_permalink(param)
|
36
|
+
# end
|
37
|
+
# end
|
38
|
+
#
|
39
|
+
def object
|
40
|
+
@object ||= end_of_association_chain.find(param) unless param.nil?
|
41
|
+
@object
|
42
|
+
end
|
43
|
+
|
44
|
+
# Used internally to load the member object in to an instance variable @#{model_name} (i.e. @post)
|
45
|
+
#
|
46
|
+
def load_object
|
47
|
+
instance_variable_set "@#{parent_type}", parent_object if parent?
|
48
|
+
instance_variable_set "@#{object_name}", object
|
49
|
+
end
|
50
|
+
|
51
|
+
# Used internally to load the collection in to an instance variable @#{model_name.pluralize} (i.e. @posts)
|
52
|
+
#
|
53
|
+
def load_collection
|
54
|
+
instance_variable_set "@#{parent_type}", parent_object if parent?
|
55
|
+
instance_variable_set "@#{object_name.to_s.pluralize}", collection
|
56
|
+
end
|
57
|
+
|
58
|
+
# Returns the form params. Defaults to params[model_name] (i.e. params["post"])
|
59
|
+
#
|
60
|
+
def object_params
|
61
|
+
params["#{object_name}"]
|
62
|
+
end
|
63
|
+
|
64
|
+
# Builds the object, but doesn't save it, during the new, and create action.
|
65
|
+
#
|
66
|
+
def build_object
|
67
|
+
@object ||= end_of_association_chain.send parent? ? :build : :new, object_params
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,59 @@
|
|
1
|
+
# Internal action lifecycle management.
|
2
|
+
#
|
3
|
+
# All of these methods are used internally to execute the options, set by the user in ActionOptions and FailableActionOptions
|
4
|
+
#
|
5
|
+
module ResourceController::Helpers::Internal
|
6
|
+
protected
|
7
|
+
# Used to actually pass the responses along to the controller's respond_to method.
|
8
|
+
#
|
9
|
+
def response_for(action)
|
10
|
+
respond_to do |wants|
|
11
|
+
options_for(action).response.each do |method, block|
|
12
|
+
if block.nil?
|
13
|
+
wants.send(method)
|
14
|
+
else
|
15
|
+
wants.send(method) { instance_eval(&block) }
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
# Calls the after callbacks for the action, if one is present.
|
22
|
+
#
|
23
|
+
def after(action)
|
24
|
+
invoke_callbacks *options_for(action).after
|
25
|
+
end
|
26
|
+
|
27
|
+
# Calls the before block for the action, if one is present.
|
28
|
+
#
|
29
|
+
def before(action)
|
30
|
+
invoke_callbacks *self.class.send(action).before
|
31
|
+
end
|
32
|
+
|
33
|
+
# Sets the flash for the action, if it is present.
|
34
|
+
#
|
35
|
+
def set_flash(action)
|
36
|
+
flash[:notice] = options_for(action).flash if options_for(action).flash
|
37
|
+
end
|
38
|
+
|
39
|
+
# Returns the options for an action, which is a symbol.
|
40
|
+
#
|
41
|
+
# Manages splitting things like :create_fails.
|
42
|
+
#
|
43
|
+
def options_for(action)
|
44
|
+
action = action == :new_action ? [action] : "#{action}".split('_').map(&:to_sym)
|
45
|
+
options = self.class.send(action.first)
|
46
|
+
options = options.send(action.last == :fails ? :fails : :success) if ResourceController::FAILABLE_ACTIONS.include? action.first
|
47
|
+
|
48
|
+
options
|
49
|
+
end
|
50
|
+
|
51
|
+
def invoke_callbacks(*callbacks)
|
52
|
+
unless callbacks.empty?
|
53
|
+
callbacks.select { |callback| callback.is_a? Symbol }.each { |symbol| send(symbol) }
|
54
|
+
|
55
|
+
block = callbacks.detect { |callback| callback.is_a? Proc }
|
56
|
+
instance_eval &block unless block.nil?
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
@@ -0,0 +1,45 @@
|
|
1
|
+
# Nested and Polymorphic Resource Helpers
|
2
|
+
#
|
3
|
+
module ResourceController::Helpers::Nested
|
4
|
+
protected
|
5
|
+
# Returns the relevant association proxy of the parent. (i.e. /posts/1/comments # => @post.comments)
|
6
|
+
#
|
7
|
+
def parent_association
|
8
|
+
@parent_association ||= parent_object.send(model_name.to_s.pluralize.to_sym)
|
9
|
+
end
|
10
|
+
|
11
|
+
# Returns the type of the current parent
|
12
|
+
#
|
13
|
+
def parent_type
|
14
|
+
@parent_type ||= [*belongs_to].find { |parent| !params["#{parent}_id".to_sym].nil? }
|
15
|
+
end
|
16
|
+
|
17
|
+
# Returns true/false based on whether or not a parent is present.
|
18
|
+
#
|
19
|
+
def parent?
|
20
|
+
!parent_type.nil?
|
21
|
+
end
|
22
|
+
|
23
|
+
# Returns the current parent param, if there is a parent. (i.e. params[:post_id])
|
24
|
+
def parent_param
|
25
|
+
params["#{parent_type}_id".to_sym]
|
26
|
+
end
|
27
|
+
|
28
|
+
# Like the model method, but for a parent relationship.
|
29
|
+
#
|
30
|
+
def parent_model
|
31
|
+
parent_type.to_s.camelize.constantize
|
32
|
+
end
|
33
|
+
|
34
|
+
# Returns the current parent object if a parent object is present.
|
35
|
+
#
|
36
|
+
def parent_object
|
37
|
+
parent? ? parent_model.find(parent_param) : nil
|
38
|
+
end
|
39
|
+
|
40
|
+
# If there is a parent, returns the relevant association proxy. Otherwise returns model.
|
41
|
+
#
|
42
|
+
def end_of_association_chain
|
43
|
+
parent? ? parent_association : model
|
44
|
+
end
|
45
|
+
end
|
@@ -0,0 +1,124 @@
|
|
1
|
+
# Thanks to Urligence, you get some free url helpers.
|
2
|
+
#
|
3
|
+
# No matter what your controller looks like...
|
4
|
+
#
|
5
|
+
# [edit_|new_]object_url # is the equivalent of saying [edit_|new_]post_url(@post)
|
6
|
+
# [edit_|new_]object_url(some_other_object) # allows you to specify an object, but still maintain any paths or namespaces that are present
|
7
|
+
#
|
8
|
+
# collection_url # is like saying posts_url
|
9
|
+
#
|
10
|
+
# Url helpers are especially useful when working with polymorphic controllers.
|
11
|
+
#
|
12
|
+
# # /posts/1/comments
|
13
|
+
# object_url #=> /posts/1/comments/#{@comment.to_param}
|
14
|
+
# object_url(comment) #=> /posts/1/comments/#{comment.to_param}
|
15
|
+
# edit_object_url #=> /posts/1/comments/#{@comment.to_param}/edit
|
16
|
+
# collection_url #=> /posts/1/comments
|
17
|
+
#
|
18
|
+
# # /products/1/comments
|
19
|
+
# object_url #=> /products/1/comments/#{@comment.to_param}
|
20
|
+
# object_url(comment) #=> /products/1/comments/#{comment.to_param}
|
21
|
+
# edit_object_url #=> /products/1/comments/#{@comment.to_param}/edit
|
22
|
+
# collection_url #=> /products/1/comments
|
23
|
+
#
|
24
|
+
# # /comments
|
25
|
+
# object_url #=> /comments/#{@comment.to_param}
|
26
|
+
# object_url(comment) #=> /comments/#{comment.to_param}
|
27
|
+
# edit_object_url #=> /comments/#{@comment.to_param}/edit
|
28
|
+
# collection_url #=> /comments
|
29
|
+
#
|
30
|
+
# Or with namespaced, nested controllers...
|
31
|
+
#
|
32
|
+
# # /admin/products/1/options
|
33
|
+
# object_url #=> /admin/products/1/options/#{@option.to_param}
|
34
|
+
# object_url(option) #=> /admin/products/1/options/#{option.to_param}
|
35
|
+
# edit_object_url #=> /admin/products/1/options/#{@option.to_param}/edit
|
36
|
+
# collection_url #=> /admin/products/1/options
|
37
|
+
#
|
38
|
+
# You get the idea. Everything is automagical! All parameters are inferred.
|
39
|
+
#
|
40
|
+
module ResourceController::Helpers::Urls
|
41
|
+
protected
|
42
|
+
['', 'edit_'].each do |type|
|
43
|
+
symbol = type.blank? ? nil : type.gsub(/_/, '').to_sym
|
44
|
+
|
45
|
+
define_method("#{type}object_url") do |*alternate_object|
|
46
|
+
smart_url *object_url_options(symbol, alternate_object.first)
|
47
|
+
end
|
48
|
+
|
49
|
+
define_method("#{type}object_path") do |*alternate_object|
|
50
|
+
smart_path *object_url_options(symbol, alternate_object.first)
|
51
|
+
end
|
52
|
+
|
53
|
+
define_method("hash_for_#{type}object_url") do |*alternate_object|
|
54
|
+
hash_for_smart_url *object_url_options(symbol, alternate_object.first)
|
55
|
+
end
|
56
|
+
|
57
|
+
define_method("hash_for_#{type}object_path") do |*alternate_object|
|
58
|
+
hash_for_smart_path *object_url_options(symbol, alternate_object.first)
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
def new_object_url
|
63
|
+
smart_url *new_object_url_options
|
64
|
+
end
|
65
|
+
|
66
|
+
def new_object_path
|
67
|
+
smart_path *new_object_url_options
|
68
|
+
end
|
69
|
+
|
70
|
+
def hash_for_new_object_url
|
71
|
+
hash_for_smart_url *new_object_url_options
|
72
|
+
end
|
73
|
+
|
74
|
+
def hash_for_new_object_path
|
75
|
+
hash_for_smart_path *new_object_url_options
|
76
|
+
end
|
77
|
+
|
78
|
+
def collection_url
|
79
|
+
smart_url *collection_url_options
|
80
|
+
end
|
81
|
+
|
82
|
+
def collection_path
|
83
|
+
smart_path *collection_url_options
|
84
|
+
end
|
85
|
+
|
86
|
+
def hash_for_collection_url
|
87
|
+
hash_for_smart_url *collection_url_options
|
88
|
+
end
|
89
|
+
|
90
|
+
def hash_for_collection_path
|
91
|
+
hash_for_smart_path *collection_url_options
|
92
|
+
end
|
93
|
+
|
94
|
+
# Used internally to provide the options to smart_url from Urligence.
|
95
|
+
#
|
96
|
+
def collection_url_options
|
97
|
+
namespaces + [parent_url_options, route_name.to_s.pluralize.to_sym]
|
98
|
+
end
|
99
|
+
|
100
|
+
# Used internally to provide the options to smart_url from Urligence.
|
101
|
+
#
|
102
|
+
def object_url_options(action_prefix = nil, alternate_object = nil)
|
103
|
+
[action_prefix] + namespaces + [parent_url_options, [route_name.to_sym, alternate_object || object]]
|
104
|
+
end
|
105
|
+
|
106
|
+
# Used internally to provide the options to smart_url from Urligence.
|
107
|
+
#
|
108
|
+
def new_object_url_options
|
109
|
+
[:new] + namespaces + [parent_url_options, route_name.to_sym]
|
110
|
+
end
|
111
|
+
|
112
|
+
def parent_url_options
|
113
|
+
parent? ? [parent_type.to_sym, parent_object] : nil
|
114
|
+
end
|
115
|
+
|
116
|
+
# Returns all of the current namespaces of the current controller, symbolized, in array form.
|
117
|
+
#
|
118
|
+
def namespaces
|
119
|
+
names = self.class.name.split("::")
|
120
|
+
names.pop
|
121
|
+
|
122
|
+
names.map(&:downcase).map(&:to_sym)
|
123
|
+
end
|
124
|
+
end
|