一直以来,对于一些选项类的操作,会设置常量和对应的描述。典型情况如下:
CONST_TO_LIST_FOR_NUMBER = [ ['新建', NEXT_NEW], ['退出', NEXT_EXIT], ['直播分机', NEXT_EXTENSION], # ['语音信箱', NEXT_RECORD], ['转人工', NEXT_HUMAN] ]
而现在更想用形式上更像声明的方式来完成这些设置,于是考虑按validates_xxx的形式完成诸如以下的调用
binary_desc :response_desc, "已回复", "未回复" const_desc :contact_type_desc, 0 => '没有操作', 1 => '录音', 2=>'SMS', 3 => 'EMAIL'
前者是针对boolean变量,后者针对有多个值的情况传入一个hash,第一个参数都是要生成的常量+函数名,常量会自动转变成大写。
binary_desc相对较简单,
src = <<-END_SRC #{desc_name.upcase} = [ ['#{true_desc}', true], ['#{false_desc}', false] ] END_SRC class_eval src unless const_defined? desc_name.upcase
需要注意的是因为true_desc会是中文,所以在#{}两侧要添加单引号。
const_desc相当要复杂些,因为在here doc里面认识外部传入的hash未找到好的办法,目前通过先转成字符串,再在eval时还原出来完成。
options_arr = (options.map {|key, content| "#{content}::#{key}"}).join('!') src = <<-END_SRC def self.#{desc_name} options_str = '#{options_arr}' options_str.split('!').map {|s| s.split('::') } end END_SRC class_eval src
join split 和map真的挺好用的。
然后要做的事情就是给这两个method找一个存放的地方。
因为很多model类会使用,所以想做成support的形式,感觉放在ActiveRecord::Base是最方便的。于是决定写个rb文件对ActiveRecord::Base进行扩展
active_record_support.rb
class ActiveRecord::Base def self.binary_desc(desc_name, true_desc, false_desc) ... end def self.const_desc(desc_name, options = {}) ... end end
起初,把这个文件放在config目录下面,同时在environment.rb的最后补充require File.join(File.dirname(__FILE__), 'active_record_support')。运行test和rails都可以成功。
但对于使用了observe的model会出问题。因为require support文件的操作在Rails::Initializer.run之后。这样的话会需要干扰到rails正常的初始化动作才能完成。于是决定还是放到model目录下面去。然后再observer处调用require 'active_record_support'。
这样,以后定义常量和对应的描述就可以更轻松一些了。