require File.expand_path(File.dirname(__FILE__) + '/../spec_helper') describe Pitch do before :each do @cases = [ { octave: 1, semitone: 0, :ratio => 2.0, :total_semitone => 12 }, { octave: 2, semitone: 0, :ratio => 4.0, :total_semitone => 24 }, { octave: 1, semitone: 6, :ratio => 2.828, :total_semitone => 18 }, { octave: 2, semitone: 6, :ratio => 5.657, :total_semitone => 30 }, { octave: 3, semitone: 7, :ratio => 11.986, :total_semitone => 43 }, { octave: -1, semitone: 0, :ratio => 0.5, :total_semitone => -12 }, { octave: -2, semitone: 0, :ratio => 0.25, :total_semitone => -24 }, { octave: -2, semitone: 7, :ratio => 0.375, :total_semitone => -17 }, { octave: -1, semitone: 9, :ratio => 0.841, :total_semitone => -3 }, ] end it "should be constructable with no parameters (no error raised)" do lambda { Pitch.new }.should_not raise_error end it "should take keyword args" do obj = Pitch.new octave: 4, semitone: 3 obj.octave.should eq(4) obj.semitone.should eq(3) end it "should use default octave and semitone if none is given" do p = Pitch.new p.ratio.should be_within(0.01).of(1.0) p.total_semitone.should eq(0) end it "should use the octave and semitone given during construction" do @cases.each do |case_data| p = Pitch.new octave: case_data[:octave], semitone: case_data[:semitone] p.ratio.should be_within(0.01).of case_data[:ratio] p.total_semitone.should be case_data[:total_semitone] end end describe '.from_ratio' do it 'should return a Pitch with given ratio' do @cases.each do |case_data| p = Pitch.from_ratio case_data[:ratio] p.octave.should eq case_data[:octave] p.semitone.should eq case_data[:semitone] p.total_semitone.should eq case_data[:total_semitone] end end end describe '.from_semitones' do it 'should return a Pitch with given total semitones' do @cases.each do |case_data| p = Pitch.from_semitones case_data[:total_semitone] p.octave.should eq case_data[:octave] p.semitone.should eq case_data[:semitone] p.total_semitone.should eq case_data[:total_semitone] end end end describe '.from_freq' do it 'should make a pitch whose freq is approximately the given freq' do [16.35, 440.0, 987.77].each do |given_freq| pitch = Pitch.from_freq given_freq pitch.freq.should be_within(0.01).of(given_freq) end end end it "should be comparable to other pitches" do p1 = Pitch.new semitone: 1 p2 = Pitch.new semitone: 2 p3 = Pitch.new semitone: 3 p1.should eq(Pitch.new semitone: 1) p2.should eq(Pitch.new semitone: 2) p3.should eq(Pitch.new semitone: 3) p1.should be < p2 p1.should be < p3 p2.should be < p3 p3.should be > p2 p3.should be > p1 p2.should be > p1 end it "should be addable and subtractable with other pitches" do p1 = Pitch.new semitone: 1 p2 = Pitch.new semitone: 2 p3 = Pitch.new semitone: 3 (p1 + p2).should eq(Pitch.new semitone: 3) (p1 + p3).should eq(Pitch.new semitone: 4) (p2 + p3).should eq(Pitch.new semitone: 5) (p1 - p2).should eq(Pitch.new semitone: -1) (p1 - p3).should eq(Pitch.new semitone: -2) (p2 - p3).should eq(Pitch.new semitone: -1) (p3 - p2).should eq(Pitch.new semitone: 1) (p3 - p1).should eq(Pitch.new semitone: 2) end it "should be addable and subtractable with integers" do p1 = Pitch.new semitone: 1 p2 = Pitch.new semitone: 2 p3 = Pitch.new semitone: 3 (p1 + 2).should eq(Pitch.new semitone: 3) (p1 + 3).should eq(Pitch.new semitone: 4) (p2 + 3).should eq(Pitch.new semitone: 5) (p1 - 2).should eq(Pitch.new semitone: -1) (p1 - 3).should eq(Pitch.new semitone: -2) (p2 - 3).should eq(Pitch.new semitone: -1) (p3 - 2).should eq(Pitch.new semitone: 1) (p3 - 1).should eq(Pitch.new semitone: 2) end it "should have freq of 440 for A4" do a4 = Pitch.new octave: 4, semitone: 9 a4.freq.should be_within(0.01).of(440.0) end describe '#to_yaml' do it 'should produce YAML that can be loaded' do p = Pitch.new(octave: 1, semitone: 2) YAML.load(p.to_yaml).should eq p end end describe '#to_s' do context 'on-letter semitones' do it 'should return the semitone letter + octave number' do { C0 => "C0", D1 => "D1", E7 => "E7", F8 => "F8", G3 => "G3", A4 => "A4", B5 => "B5", C2 => "C2" }.each do |p,s| p.to_s.should eq s end end end context 'off-letter semitones' do context 'sharpit set false' do it 'should return semitone letter + "b" + octave number' do { Db0 => "Db0", Eb1 => "Eb1", Gb7 => "Gb7", Ab4 => "Ab4", Bb1 => "Bb1" }.each do |p,s| p.to_s(false).should eq s end end end context 'sharpit set true' do it 'should return semitone letter + "#" + octave number' do { Db0 => "C#0", Eb1 => "D#1", Gb7 => "F#7", Ab4 => "G#4", Bb1 => "A#1" }.each do |p,s| p.to_s(true).should eq s end end end end end end