菜鸟教程小白 发表于 2022-12-12 14:39:54

ios - AudioKit ios AKSequencer 无法准确重启播放


                                            <p><p>我正在尝试使用 AudioKit 在小节的每个节拍上播放声音。虽然我已经实现了 <a href="https://stackoverflow.com/questions/47629637/audiokit-ios-aksamplermetronome/47630809#47630809" rel="noreferrer noopener nofollow">this</a> 中的代码关于通过 AudioKit 进行回调的类似问题,我似乎无法让音序器正确更新更改和播放。它会准确播放一次,但是在倒带和更改值后,它将仅使用初始值(或根本不播放)。</p>

<p>我的目的是为每个小节创建一个带有节拍值的小节结构,然后使用 MIDI 和回调根据有多少小节/节拍播放不同的声音。谢谢!</p>

<pre><code>import UIKit
import AudioKit

class ViewController: UIViewController {

let sequencer = AKSequencer()

let click = AKSynthSnare()
let callbackInst = AKCallbackInstrument()

// Create the struct that defines each line
struct Line {
    var name: String
    var measures: Int
    var beatsPerMeasure: Int
    func totalBeats() -&gt; Int {
      return (measures * beatsPerMeasure)
    }
}

// Initialize intro line
var intro = Line(name: &#34;Intro&#34;, measures: 0, beatsPerMeasure: 0)

// A function to create/update/playback the sequence on button press
func playBack() {

    let metronomeTrack = sequencer.newTrack()
    metronomeTrack?.setMIDIOutput(click.midiIn)
    let callbackTrack = sequencer.newTrack()
    callbackTrack?.setMIDIOutput(callbackInst.midiIn)

    for steps in 0 ... Int(measuresRowOneValue) {
      // this will trigger the sampler on the four down beats
      metronomeTrack?.add(noteNumber: 60, velocity: 100, position: AKDuration(beats: Double(steps)), duration: AKDuration(beats: 0.5))

      // set the midiNote number to the current beat number
      callbackTrack?.add(noteNumber: MIDINoteNumber(steps), velocity: 100, position: AKDuration(beats: Double(steps)), duration: AKDuration(beats: 0.5))

      // set the callback
      callbackInst.callback = {status, noteNumber, velocity in
            guard status == .noteOn else { return }
            print(&#34;beat number: \(noteNumber + 1)&#34;)

      }
    }
}


@IBOutlet weak var rowOneLocationOne: UIImageView!
// Listener for UI display values
var measuresRowOneValue: Int = 0 {
    didSet {
      intro.measures = measuresRowOneValue
    }
}

@IBAction func rowOnePlusButton(_ sender: UIButton) {
    measuresRowOneValue += 1
}

@IBAction func rowOneMinusButton(_ sender: UIButton) {
    measuresRowOneValue -= 1
}

@IBAction func playbackStart(_ sender: UIButton) {
    playBack()
    sequencer.play()
}

@IBAction func playbackStop(_ sender: UIButton) {
    sequencer.stop()
}

@IBAction func playbackRestart(_ sender: UIButton) {
    sequencer.rewind()
}

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib.

    AudioKit.output = click
    try!AudioKit.start()
}
}
</code></pre></p>
                                    <br><hr><h1><strong>Best Answer-推荐答案</ strong></h1><br>
                                            <p><p>您的代码中有一些令人困惑的事情,所以我不确定这是否是您唯一的问题,但至少,每次您更改序列的长度时,您都需要调用 <code>setLength() </code> 后跟 <code>enableLooping</code>。基本上,默认情况下(即,除非您明确设置长度)序列的长度将是序列中最长轨道的长度。在您的“播放”方法中,您在轨道顶部添加轨道而不删除旧轨道,因此它无法知道您希望序列持续多长时间。</p>

<p>您的“播放”方法正在做两件不同的事情(都不涉及播放)。你可能想打破它。你可以有一个 <code>setup()</code> 来做只需要做一次的事情(创建轨道,设置它们的输出,设置回调)和一个 <code>rewriteSequence()</code> 方法,当你想重写轨道时被调用。通过这种方式,您可以重复使用现有轨道,而不是不断创建新轨道。</p>

<pre><code>var metronomeTrack: AKMusicTrack!
var callbackTrack: AKMusicTrack!

    // call this just once at the start
    func setup() {
      metronomeTrack = sequencer.newTrack()
      metronomeTrack?.setMIDIOutput(click.midiIn)
      callbackTrack = sequencer.newTrack()
      callbackTrack?.setMIDIOutput(callbackInst.midiIn)

      callbackInst.callback = {status, noteNumber, velocity in
            guard status == .noteOn else { return }
            print(&#34;beat number: \(noteNumber + 1)&#34;)

      }
    }

    // call this each time you want to change the sequence
    func rewriteSequence() {
      metronomeTrack?.clear()
      callbackTrack?.clear()
      for steps in 0 ... Int(measuresRowOneValue) {
            metronomeTrack?.add(noteNumber: 60, velocity: 100, position: AKDuration(beats: Double(steps)), duration: AKDuration(beats: 0.5))
            callbackTrack?.add(noteNumber: MIDINoteNumber(steps), velocity: 100, position: AKDuration(beats: Double(steps)), duration: AKDuration(beats: 0.5))
      }

      // This will make sure it loops correctly:
      sequencer.setLength(AKDuration(beats: Double(measuresRowOneValue)))
      sequencer.enableLooping()
    }
</code></pre>

<p>我希望这会有所帮助。</p></p>
                                   
                                                <p style="font-size: 20px;">关于ios - AudioKit ios AKSequencer 无法准确重启播放,我们在Stack Overflow上找到一个类似的问题:
                                                        <a href="https://stackoverflow.com/questions/50622527/" rel="noreferrer noopener nofollow" style="color: red;">
                                                                https://stackoverflow.com/questions/50622527/
                                                        </a>
                                                </p>
                                       
页: [1]
查看完整版本: ios - AudioKit ios AKSequencer 无法准确重启播放