Sha256: 91be67ae5c5c9c4cb70890cf685928bfe905137a99f8f450785dc7b417a8a5d8

Contents?: true

Size: 1.37 KB

Versions: 7

Compression:

Stored size: 1.37 KB

Contents

# frozen_string_literal: true

# A harmonic interval is the diatonic interval between two notes sounding together.
class HeadMusic::HarmonicInterval
  attr_reader :voice1, :voice2, :position

  def initialize(voice1, voice2, position)
    @voice1 = voice1
    @voice2 = voice2
    @position = position.is_a?(String) ? HeadMusic::Content::Position.new(voice1.composition, position) : position
  end

  def diatonic_interval
    @diatonic_interval ||= HeadMusic::DiatonicInterval.new(lower_pitch, upper_pitch)
  end

  def voices
    [voice1, voice2].compact
  end

  def notes
    @notes ||= voices.map { |voice| voice.note_at(position) }.compact.sort_by(&:pitch)
  end

  def lower_note
    notes.first
  end

  def upper_note
    notes.last
  end

  def pitches
    @pitches ||= notes.map(&:pitch).sort_by(&:to_i)
  end

  def lower_pitch
    pitches.first
  end

  def upper_pitch
    pitches.last
  end

  def pitch_orientation
    return if lower_pitch == upper_pitch

    if lower_note.voice == voice1
      :up
    elsif lower_note.voice == voice2
      :down
    end
  end

  def to_s
    "#{diatonic_interval} at #{position}"
  end

  def method_missing(method_name, *args, &block)
    respond_to_missing?(method_name) ? diatonic_interval.send(method_name, *args, &block) : super
  end

  def respond_to_missing?(method_name, *_args)
    diatonic_interval.respond_to?(method_name)
  end
end

Version data entries

7 entries across 7 versions & 1 rubygems

Version Path
head_music-4.0.1 lib/head_music/harmonic_interval.rb
head_music-4.0.0 lib/head_music/harmonic_interval.rb
head_music-3.0.1 lib/head_music/harmonic_interval.rb
head_music-3.0.0 lib/head_music/harmonic_interval.rb
head_music-2.0.1 lib/head_music/harmonic_interval.rb
head_music-2.0.0 lib/head_music/harmonic_interval.rb
head_music-1.0.0 lib/head_music/harmonic_interval.rb