我正在尝试创建一个灵活的 dsl。我已经有了 DSL 模块,比如说module DSL
。DSL 用户可以将其作为一个类创建衍生产品。DSL 的要点是允许用户Feature
使用自定义渲染方法创建对象。有很多丑陋且非 DRY 的代码支持 . Feature
,因此是抽象的,但是用户需要对该功能的呈现方式进行大量控制,而我的元编程无法胜任这项任务。让我告诉你它是如何设置的。
DSL 看起来像这样:
module DSL
module ClassMethods
attr_accessor :features
def column(name, *args)
arguments = args.pop || {}
self.features = [] if self.features.nil?
self.features << Feature.new(name, arguments)
end
end
def self.included(base)
base.extend ClassMethods
end
end
end
它的实现看起来像这样:
class DSLSpinOff
include DSL
feature :one
feature :two, render_with: :predefined_render
feature :three, render_with: :user_defined_render
feature :four, render_with: lambda {
puts "Go nuts, user!"
puts "Do as you please!"
}
def user_defined_render
#...
end
end
最后,要素类本身位于 DSL 中,如下所示:
module DSL
#...
private
class Feature
attr_accessor :name, :render_with
def initialize(name, *args)
self.name = name
attributes = args.pop || {}
# somehow delegate attributes[:render_with] to the render function, handling defaults, lamdbas, function string names, etc
self.render_with = attributes.fetch(:render_with, :default_render)
end
private
def default_render
#...
end
def predefined_render
#...
end
end
end