Sha256: 5ef57f3c1efbc348f18320f96703c7cff2b0afec12176dd932ab3eefb01a8fc4

Contents?: true

Size: 1.97 KB

Versions: 1

Compression:

Stored size: 1.97 KB

Contents

require "wavefile"
require "mml2wav/scale"

module Mml2wav
  class Wave
    class << self
      include WaveFile

      def write(sounds, options={})
        output_path = options[:output] || "doremi.wav"
        sampling_rate = options[:sampling_rate] || 22050
        bpm = options[:bpm] || 120
        velocity = 5
        octave = 4
        default_length = 4.0

        format = Format.new(:mono, :pcm_8, sampling_rate)
        Writer.new(output_path, format) do |writer|
          buffer_format = Format.new(:mono, :float, sampling_rate)
          sounds.scan(/T\d+|V\d+|L\d+|[A-G][#+-]?\d*\.?|O\d+|[><]|./i).each do |sound|
            base_sec = 60.0 * 4
            length = default_length
            case sound
            when /\AT(\d+)/i
              bpm = $1.to_i
            when /\AV(\d+)/i
              velocity = $1.to_i
            when /\AL(\d+)/i
              default_length = $1.to_f
            when /\A([A-G][#+-]?)(\d+)(\.)?/i
              length = $2.to_f
              sound = $1
              length = default_length / 1.5 if $3
            when /\AO(\d+)/i
              octave = $1.to_i
            when "<"
              octave += 1
            when ">"
              octave -= 1
            end
            sec = base_sec / length / bpm
            amplitude = velocity.to_f / 10
            frequency = Scale::FREQUENCIES[sound.downcase]
            next unless frequency
            frequency *= (2 ** octave)
            samples = sine_wave(frequency, sampling_rate, sec, amplitude)
            buffer = Buffer.new(samples, buffer_format)
            writer.write(buffer)
          end
        end
      end

      private
      def sine_wave(frequency, sampling_rate, sec, amplitude=0.5)
        max = sampling_rate * sec
        if frequency == 0
          return Array.new(max) { 0.0 }
        end
        base_x = 2.0 * Math::PI * frequency / sampling_rate
        1.upto(max).collect do |n|
          amplitude * Math.sin(base_x * n)
        end
      end
    end
  end
end

Version data entries

1 entries across 1 versions & 1 rubygems

Version Path
mml2wav-0.0.3 lib/mml2wav/wave.rb