在我的Rails应用程序中,ApplicationController中定义了一个我需要在模块中使用的方法。所有控制器/助手也需要访问该模块。我不确定让模块访问此方法的最佳方法是什么。我认为有一种方法可以通过包装模块然后将其包含在ApplicationController中来实现。
我还想保留命名空间,而不是将模块方法直接包含在控制器中。例如controller.Module.foo而不仅仅是controller.foo
这就是我要做的事情:
module Analytics
module Events
extend self
def create_event name
# do some stuff
controller_method name
end
end
end
class ApplicationController < ActionController::Base
include Analytics
def controller_method
# ...
end
def some_other_method
Events.create_event :test
end
end
现在我知道我可以在Analytics模块中添加一个self.included方法,它给我传入的基础对象,但我不知道如何在KissMetrics模块中使用该基础。
这可能还是有更好的方法吗?
为了进一步澄清,我正在使用一个扩展/包含在ActionController :: Base中的gem。请参阅此script的底部。然后在ApplicationController中配置该库。我想做的就是拥有一个模块,其中包含使用此gem的已定义事件。这使我更容易在所有控制器上保持一致的事件,例如
module Events
def sign_up_event
analytical.event "sign up" # calls the library that was loaded into the app controller
end
end
# in some controller
Events.sign_up_event
# instead of this, which is totally usable but not DRY
analytical.event "sign up"
答案 0 :(得分:0)
据我所知,你不应该从lib调用控制器动作,反之亦然。
我建议创建一个库,因为我不确切地知道你要构建什么我已经设置了一个例子:
lib / analytics.rb中的
class Analytics
def initialize(object_id, event_id)
# insert object to event relation logic here
@object = Object.find(object_id)
@event = Event.find(event_id)
end
def add_instance_method
# add logic for instantiated Analytics object
@object.event_collection.create(event_id: @event.id)
end
def self.check_class_method?(object_id, event_id)
# check logic for Analytic Class taking 2 arguments
@object = Object.find(object_id)
@event = event.find(event_id)
@object.events.include?(@event)
# return true/false
end
end
所以现在在你的控制器中:
def action
@object = Object.find(params[:id])
@event = Event.find_by_identifier("logged_in")
unless Analytics.check_class_method?(@object_id, @event_id)
analytic = Analytics.new(@object.id, @event.id)
raise "Failed" unless analytic.add_instance_method
end
end
答案 1 :(得分:0)
决定通过将模块更改为类并在ApplicationController中的before_filter回调中实例化它来解决此问题。我把初始化的分析对象传递给它,一切都很好。本来希望使用一个模块但是这样可以解决问题。