**DrumThumper** ========== Oboe playback sample app. ## Abstract **DrumThumper** is a "Drum Pad" app which demonstrates best-practices for low-latency audio playback using the Android **Oboe** API. **DrumThumper** consists of a set of trigger pad widgets and an optional UI for controlling the level and stereo placement of each of the virtual drums. The audio samples are stored in application resources as WAV data. This is parsed and loaded (by routines in **parselib**) into memory blocks. The audio samples are mixed and played by routines in **iolib**. **DrumThumper** is written in a combination of Kotlin for the UI and JNI/C++ for the player components (to demonstrate accessing native code from a Kotlin or Java application). ## Scope **DrumThumper** is designed with the following goals in mind: * To demonstrate the most efficient means of playing audio with the lowest possible latency. * To demonstrate how to play multiple sources of audio mixed into a single Oboe stream. * To demonstrate the life-cycle of an Oboe audio stream and it's relationship to the application lifecycle. * To demonstrate the correct handling of playback errors and output device connection/disconnection. Secondarily, **DrumThumper** demonstrates: * Using Android "assets" for audio data. * The mechanism for calling native (C/C++) audio functionality from a Kotlin/Java app. * A mechanism for sharing binary data between a Kotlin/Java app with the native (C/C++) layer. * A mechanism for parsing/loading one type (WAV) of audio data. * How to control the relative levels (gain) of audio sources mixed into an output stream. * How to locate a mono data source in a stereo output stream. * How to use the Oboe resampler to resample source audio to the device playback rate, and therefore not incur this overhead at playback time. To keep things simple, **DrumThumper** specifically does not: * Does not support audio samples in other than 16-bit, mono PCM Samples. It does not support Stereo or different PCM formats. * Does not support non-WAV audio data (such as AIFF). * Does not support compressed audio data. **DrumThumper** now supports different sample rates for the source samples. If one wanted to extend **DrumThumper** to support Stereo samples: * The SampleSource class would need to be extended to understand Stereo SampleBuffer objects, it currently assumes Mono. * The OneShotSampleSource.mixAudio() method would need to have separate mixing logic for Stereo and Mono SampleSource. ## DrumThumper project structure ### Kotlin App Layer Contains classes for the application logic and defines the methods for accessing the native data and player functionality. ### Native (C++) layer Contains the implementation of the `native` (JNI) methods defined in the `DrumPlayer` (Kotlin) class. ### Dependent Libraries * **iolib** Classes for playing audio data. * **parselib** Classes for parsing and loading audio data from WAV resources. ## App * DrumPlayer.kt The Kotlin class which provides the audio playback functionality by interfacing with the native (C++) libraries. * DrumThumperActivity.kt The main application logic. * TriggerPad.kt An Android View subclass which implements the "trigger pad" UI widgets ## Native-interface (JNI) * DrumPlayerJNI.cpp This is where all the access to the native functionality is implemented.