File: onsets.rb

package info (click to toggle)
ruby-aubio 0.3.6-3
  • links: PTS, VCS
  • area: main
  • in suites: bookworm, sid, trixie
  • size: 7,236 kB
  • sloc: ruby: 420; sh: 4; makefile: 3
file content (73 lines) | stat: -rw-r--r-- 2,077 bytes parent folder | download | duplicates (2)
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