Note.new(2, 43, 0.25, 127, now += interval)
This would be much nicer:
Note.new(:channel => 2,
:note => 43,
:duration => 0.25,
:velocity => 127,
:time => now += interval)
(God I wish I could configure bloody Gmail to use monospaced fonts.)
Anyway, is there an easy equivalent to Struct which, instead of taking
stuff in sequence, takes a Hash of options, in a Rails-y style? Like a
HashStruct? Does Facets have such a thing, maybe haps? If not, is the
basic code for Struct in Ruby, and/or is a class like Struct but with
Hashes easy to build?
--
Giles Bowkett
Podcast: http://hollywoodgrit.blogspot.com
Blog: http://gilesbowkett.blogspot.com
Portfolio: http://www.gilesgoatboy.org
Tumblelog: http://giles.tumblr.com
> Anyway, is there an easy equivalent to Struct which, instead of taking
> stuff in sequence, takes a Hash of options, in a Rails-y style? Like a
> HashStruct? Does Facets have such a thing, maybe haps? If not, is the
> basic code for Struct in Ruby, and/or is a class like Struct but with
> Hashes easy to build?
i use this alot in rails projects:
require 'attributes'
class OpenObject
alias_method "__inspect__", "inspect"
alias_method "__to_s__", "to_s"
alias_method "__instance_eval__", "instance_eval"
instance_methods.each{|m| undef_method m unless m[%r/__/]}
alias_method "instance_eval", "__instance_eval__"
alias_method "inspect", "__inspect__"
alias_method "to_s", "__to_s__"
def method_missing m, *a, &b
super if(a.empty? and b.nil?)
m = m.to_s
setter = m.delete!('=') || a.first || b
if setter
if a.empty?
__attribute__ m, &b
else
__attribute__ m => a.shift
end
else
nil
end
end
def configure kvs = {}, &b
kvs.each{|k,v| __attribute__ k => v}
__instance_eval__ &b if b
end
alias_method "initialize", "configure"
alias_method "eval", "configure"
def __singleton_class__ &b
sc =
class << self
self
end
b ? sc.module_eval(&b) : sc
end
def extend *a, &b
if b
__singleton_class__ &b
self
else
super
end
end
alias_method "extending", "extend"
def attributes *a, &b
__attributes__ *a, &b
end
def to_hash
__attributes__.inject(Hash.new){|h,a| h.update a => __send__(a)}
end
def as_query
require 'cgi'
e = lambda{|x| CGI.escape x.to_s}
to_hash.map{|k,v| "#{ e[k] }=#{ e[v] }"}.join("&")
end
def respond_to? *a, &b
true
end
end
def openobject(*a, &b) OpenObject.new(*a, &b) end
class Object
def Open *a, &b
OpenObject.new(*a, &b)
end
end
it's all that and so much more.
a @ http://codeforpeople.com/
--
share your knowledge. it's a way to achieve immortality.
h.h. the 14th dalai lama
One possibility, method for Note only:
manveru@pi ~ % irb
class Note < Struct.new(:channel, :note, :duration, :velocity, :time)
def self.create(hash)
new(*hash.values_at(*members.map{|m| m.to_sym}))
end
end
# nil
Note.create(:channel => 1, :note => 2, :duration => 3, :velocity => 3,
:time => Time.now)
# #<struct Note channel=1, note=2, duration=3, velocity=3, time=Wed
Jan 09 14:36:21 +0900 2008>
Second one, method for all Structs:
manveru@pi ~ % irb
class Struct
def self.create(hash)
new(*hash.values_at(*members.map{|m| m.to_sym}))
end
end
# nil
Note = Struct.new(:channel, :note, :duration, :velocity, :time)
# Note
Note.create(:channel => 1, :note => 2, :duration => 3, :velocity => 3,
:time => Time.now)
# #<struct Note channel=1, note=2, duration=3, velocity=3, time=Wed
Jan 09 14:37:20 +0900 2008>
Then just use that form.
Kind regards
robert
--
use.inject do |as, often| as.you_can - without end
> I have some code which uses Structs. I'd rather use keyword args. I'm
> generating data objects to store MIDI note data, and it's kind of
> hideously unreadable:
>
> Note.new(2, 43, 0.25, 127, now += interval)
>
> This would be much nicer:
>
> Note.new(:channel => 2,
> :note => 43,
> :duration => 0.25,
> :velocity => 127,
> :time => now += interval)
One way is:
require "ostruct"
now = Time.now
interval = 5
note = OpenStruct.new( :channel => 2,
:note => 43,
:duration => 0.25,
:velocity => 127,
:time => now + interval )
note.channel # => 2
note.velocity # => 127
__END__
James Edward Gray II
> Second one, method for all Structs:
>
> manveru@pi ~ % irb
> class Struct
> def self.create(hash)
> new(*hash.values_at(*members.map{|m| m.to_sym}))
> end
> end
> # nil
> Note = Struct.new(:channel, :note, :duration, :velocity, :time)
> # Note
> Note.create(:channel => 1, :note => 2, :duration => 3, :velocity => 3,
> :time => Time.now)
> # #<struct Note channel=1, note=2, duration=3, velocity=3, time=Wed
> Jan 09 14:37:20 +0900 2008>
Ara's seems the most robust, but with so much going on and no specs it
makes me antsy; the OpenStruct looks good too but I want classes as
well as instances.
Folks, why do you run around implementing something? The desired
behavior *is already present* with Struct.
Cheers
robert
I tried your code example, and it didn't work.
class Note < Struct.new(:channel) ; end
Note.new(:channel => 2)
Note.channel # => "{:channel => 2}"
It just stores the hash.
Argh!! You are right, I am wrong. I'm sorry, should have payed
attention more closely to the output of #inspect. I think I'll better
shut up now.
Cheers
robert
Hm, could not resist... Here is another variant:
Note = Struct.new(:channel, :note, :duration, :velocity, :time) do
def self.create(hash)
new(*members.map {|me| hash[me.to_sym]})
end
end
irb(main):016:0* now = interval = 1
=> 1
irb(main):017:0> n = Note.new(:channel => 2,
irb(main):018:1* :note => 43,
irb(main):019:1* :duration => 0.25,
irb(main):020:1* :velocity => 127,
irb(main):021:1* :time => now += interval)
=> #<struct Note channel={:channel=>2, :note=>43, :duration=>0.25,
:velocity=>127, :time=>2}, note=nil, duration=nil, ve
locity=nil, time=nil>
irb(main):022:0>
:-)