1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
|
# frozen_string_literal: true
module Aubio
class Onsets
def initialize(aubio_source, params)
# TODO: cleanup param dups
@sample_rate = params[:sample_rate] || 44_100
@window_size = params[:window_size] || 1024
@hop_size = params[:hop_size] || 512
@source = aubio_source
@onset = Api.new_aubio_onset('default', @window_size, @hop_size, @sample_rate)
Api.aubio_onset_set_minioi_ms(@onset, 12.0)
Api.aubio_onset_set_threshold(@onset, 0.3)
# create output for source
@sample_buffer = Api.new_fvec(@hop_size)
# create output for pitch and beat
@out_fvec = Api.new_fvec(1)
end
def each
return enum_for(:each) unless block_given?
total_frames_counter = 0
read_buffer = FFI::MemoryPointer.new(:int)
loop do
# Perform onset calculation
Api.aubio_source_do(@source, @sample_buffer, read_buffer)
Api.aubio_onset_do(@onset, @sample_buffer, @out_fvec)
# Retrieve result
onset_new_peak = Api.fvec_get_sample(@out_fvec, 0)
no_of_bytes_read = read_buffer.read_int
total_frames_counter += no_of_bytes_read
if onset_new_peak > 0.0
onset_seconds = Api.aubio_onset_get_last_s(@onset)
onset_milliseconds = Api.aubio_onset_get_last_ms(@onset)
output = {
s: onset_seconds,
ms: onset_milliseconds,
start: (onset_seconds == 0.0 ? 1 : 0),
end: 0
}
yield output
end
next unless no_of_bytes_read != @hop_size
# there's no more audio to look at
# Let's output one last onset to mark the end of the file
total_time = total_frames_counter.to_f / @sample_rate.to_f
output = {
s: total_time,
ms: total_time / 1000.0,
start: 0,
end: 1
}
yield output
# clean up
Api.del_aubio_onset(@onset)
Api.del_fvec(@sample_buffer)
Api.del_fvec(@out_fvec)
break
end
end
end
end
|