我可以在包含和扩展Ruby模块和类时使用一些帮助。
由于default_url_options
Hash,我之前的问题处理了命名路由,但并未处理所有视图/标记助手。这里的问题是ActionController :: UrlWriter方法,如url_for
,调用类属性default_url_options
。因此,当包含ActionController :: UrlWriter时,它会扩展当前的类单例,但也需要扩展当前类本身。如果你看下面我的代码,MyBigClass应该在它的类上有default_url_options,而不是实例。这有效,但我不确定它是否正确或可能会破坏某些东西。
这是我目前的模块:
module MessageViewHelper
module Methods
def self.included(base)
base.module_eval do
include TemplatesHelper
include LegacyUrlsHelper
include ActionView::Helpers::TagHelper
include ActionView::Helpers::AssetTagHelper
include ActionView::Helpers::UrlHelper
include ActionController::UrlWriter
end
MyBigClass.class_eval do
cattr_accessor :default_url_options
def self.default_url_options(options = {})
options.merge({:host=>'www.myhostname.com'})
end
end
unless ActionController::UrlWriter.method_defined?('old_url_for')
ActionController::UrlWriter.class_eval do
alias_method :old_url_for, :url_for
def url_for(options)
if options.is_a?(String)
options
else
old_url_for(options)
end
end
end
end # unless method_defined?
end
end
end
class MyBigClass < ActiveRecord::Base
def message(template_name)
class << self; include MessageViewHelper::Methods; end
# ... more stuff here
end
end
我知道我对ruby类/模块设计和扩展并不完全清楚。有没有人对此有任何见解? MyBigClass上的更改是否应在message
结束时恢复?
答案 0 :(得分:0)
从类方法或class_eval块中调用include将把包含的模块的定义带入类本身。我并不完全明白你要做什么,但根据你的描述,我认为你正确地做到了。
答案 1 :(得分:0)
MyBigClass.class_eval do
cattr_accessor :default_url_options
def self.default_url_options(options = {})
options.merge({:host=>'www.myhostname.com'})
end
end
通常,class_eval在类上定义常规实例方法。
但是,在这种情况下,在class_eval块中你定义了自我的方法。块内的self是MyBigClass,因此它实际上会在MyBigClass的singleton类中创建一个方法。单例类中的任何方法都不是实例方法。