def fattrs *a, &b
unless a.empty?
returned = Hash.new
hashes, names = a.partition{|x| Hash === x}
names_and_defaults = {}
hashes.each{|h| names_and_defaults.update h}
names.flatten.compact.each{|name| names_and_defaults.update name => nil}
initializers = __fattrs__.initializers
names_and_defaults.each do |name, default|
raise NameError, "bad instance variable name '@#{ name }'" if "@#{ name }" =~ %r/[!?=]$/o
name = name.to_s
initialize = b || lambda { default }
initializer = lambda do |this|
Object.instance_method('instance_eval').bind(this).call &initialize
end
initializer_id = initializer.object_id
__fattrs__.initializers[name] = initializer
compile = lambda do |code|
begin
module_eval code
rescue SyntaxError
raise SyntaxError, "\n#{ code }\n"
end
end
code = "def \#{ name }=(*value, &block)\nvalue.unshift block if block\n@\#{ name } = value.first\nend\n"
compile[code]
code = "def \#{ name }(*value, &block)\nvalue.unshift block if block\nreturn self.send('\#{ name }=', value.first) unless value.empty?\n\#{ name }! unless defined? @\#{ name }\n@\#{ name }\nend\n"
compile[code]
code = "def \#{ name }!\ninitializer = ObjectSpace._id2ref \#{ initializer_id }\nself.\#{ name } = initializer.call(self)\n@\#{ name }\nend\n"
compile[code]
code = "def \#{ name }?\nself.\#{ name }\nend\n"
compile[code]
fattrs << name
returned[name] = initializer
end
returned
else
begin
__fattr_list__
rescue NameError
singleton_class =
class << self
self
end
klass = self
singleton_class.module_eval do
fattr_list = List.new
define_method('fattr_list'){ klass == self ? fattr_list : raise(NameError) }
alias_method '__fattr_list__', 'fattr_list'
end
__fattr_list__
end
end
end