#ifndef Magnum_Audio_Source_h #define Magnum_Audio_Source_h /* This file is part of Magnum. Copyright © 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018 Vladimír Vondruš Copyright © 2015 Jonathan Hale Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /** @file * @brief Class @ref Magnum::Audio::Source */ #include #include #include #include #include "Magnum/Magnum.h" #include "Magnum/Audio/Audio.h" #include "Magnum/Audio/visibility.h" #include "Magnum/Math/Vector3.h" namespace Magnum { namespace Audio { /** @brief Source Manages positional audio source. @todo Expose convenient API for buffer queuing */ class MAGNUM_AUDIO_EXPORT Source { public: /** * @brief Constructor * * Creates OpenAL source object. * @see @fn_al_keyword{GenSources} */ explicit Source() { alGenSources(1, &_id); } /** * @brief Destructor * * Deletes OpenAL source object. * @see @fn_al_keyword{DeleteSources} */ ~Source() { if(_id) alDeleteSources(1, &_id); } /** @brief Copying is not allowed */ Source(const Source&) = delete; /** @brief Move constructor */ Source(Source&& other); /** @brief Copying is not allowed */ Source& operator=(const Source&) = delete; /** @brief Move assignment */ Source& operator=(Source&& other); /** @brief OpenAL source ID */ ALuint id() const { return _id; } /** @{ @name Source positioning */ /** * @brief Position * @see @fn_al_keyword{GetSourcefv} with @def_al{POSITION} */ Vector3 position() const { Vector3 v; alGetSourcefv(_id, AL_POSITION, v.data()); return v; } /** * @brief Set position * @return Reference to self (for method chaining) * * Default is @cpp {0.0f, 0.0f, 0.0f} @ce. * @see @ref setRelative(), @fn_al_keyword{Sourcefv} with * @def_al{POSITION} */ Source& setPosition(const Vector3& position) { alSourcefv(_id, AL_POSITION, position.data()); return *this; } /** @overload * @see @fn_al_keyword{Sourceiv} with @def_al{POSITION} */ Source& setPosition(const Vector3i& position) { alSourceiv(_id, AL_POSITION, position.data()); return *this; } /** * @brief Velocity * @see @fn_al_keyword{GetSourcefv} with @def_al{VELOCITY} */ Vector3 velocity() const { Vector3 v; alGetSourcefv(_id, AL_VELOCITY, v.data()); return v; } /** * @brief Set velocity * @return Reference to self (for method chaining) * * Default is @cpp {0.0f, 0.0f, 0.0f} @ce. * @see @ref setRelative(), @fn_al_keyword{Sourcefv} with * @def_al{VELOCITY} */ Source& setVelocity(const Vector3& velocity) { alSourcefv(_id, AL_VELOCITY, velocity.data()); return *this; } /** @overload * @see @fn_al_keyword{Sourceiv} with @def_al{VELOCITY} */ Source& setVelocity(const Vector3i& velocity) { alSourceiv(_id, AL_VELOCITY, velocity.data()); return *this; } /** * @brief Whether the source is interpreted relative to the listener * @see @ref position(), @ref direction(), @ref velocity(), * @fn_al_keyword{GetSourcei} with @def_al{SOURCE_RELATIVE} */ bool isRelative() const { Int relative; alGetSourcei(_id, AL_SOURCE_RELATIVE, &relative); return (relative == 1); } /** * @brief Interpret source relatively to listener * * When enabled, source position, direction and velocity will be * interpreted relatively to listener. Default is @cpp false @ce. * @see @ref setPosition(), @ref setDirection(), @ref setVelocity(), * @fn_al_keyword{Sourcei} with @def_al{SOURCE_RELATIVE} */ Source& setRelative(bool relative) { alSourcei(_id, AL_SOURCE_RELATIVE, relative); return *this; } /*@}*/ /** @{ @name Source behavior */ /** * @brief Gain * @see @fn_al_keyword{GetSourcef} with @def_al{GAIN} */ Float gain() const { Float gain; alGetSourcef(_id, AL_GAIN, &gain); return gain; } /** * @brief Set gain * @return Reference to self (for method chaining) * * Default is @cpp 1.0f @ce, which means that the sound is * unattenuated. If set to @cpp 0.0f @ce, the source is muted. * @see @ref setMinGain(), @ref setMaxGain(), @fn_al_keyword{Sourcef} * with @def_al{GAIN} */ Source& setGain(Float gain) { alSourcef(_id, AL_GAIN, gain); return *this; } /** * @brief Minimal gain to clamp to * @see @ref setMaxGain(), @ref setGain(), @ref maxGain(), * @ref gain(), @fn_al_keyword{GetSourcef} with @def_al{MIN_GAIN} */ Float minGain() const { Float minGain; alGetSourcef(_id, AL_MIN_GAIN, &minGain); return minGain; } /** * @brief Set min gain * @return Reference to self (for method chaining) * * If effective gain is lower than min gain, min gain is used. Note * that this is done before listener gain is applied. Default is * @cpp 0.0f @ce. * @see @ref setMaxGain(), @ref setGain(), @fn_al_keyword{Sourcef} with * @def_al{MIN_GAIN} */ Source& setMinGain(Float gain) { alSourcef(_id, AL_MIN_GAIN, gain); return *this; } /** * @brief Maximal gain to clamp to * @see @ref setMinGain(), @ref setGain(), @ref maxGain(), * @ref gain(), @fn_al_keyword{GetSourcef} with @def_al{MAX_GAIN} */ Float maxGain() const { Float maxGain; alGetSourcef(_id, AL_MAX_GAIN, &maxGain); return maxGain; } /** * @brief Set max gain * @return Reference to self (for method chaining) * * If effective gain is higher than max gain, max gain is used. Note * that this is done before listener gain is applied. Default is * @cpp 1.0f @ce. If set to @cpp 0.0f @ce, the source is muted. * @see @ref setMinGain(), @ref setGain(), @fn_al_keyword{Sourcef} with * @def_al{MIN_GAIN} */ Source& setMaxGain(Float gain) { alSourcef(_id, AL_MAX_GAIN, gain); return *this; } /** * @brief Reference distance * @see @fn_al_keyword{GetSourcef} with @def_al{REFERENCE_DISTANCE} */ Float referenceDistance() const { Float distance; alGetSourcef(_id, AL_REFERENCE_DISTANCE, &distance); return distance; } /** * @brief Set reference distance * @return Reference to self (for method chaining) * * Default is @cpp 1.0f @ce. Distance at which the listener will * experience @ref gain() (or @ref minGain(), @ref maxGain() * if gain was clamped). * @see @ref setRolloffFactor(), @fn_al_keyword{Sourcef} with * @def_al{REFERENCE_DISTANCE} */ Source& setReferenceDistance(Float distance) { alSourcef(_id, AL_REFERENCE_DISTANCE, distance); return *this; } /** @overload * @see @fn_al_keyword{Sourcei} with @def_al{REFERENCE_DISTANCE} */ Source& setReferenceDistance(Int distance) { alSourcei(_id, AL_REFERENCE_DISTANCE, distance); return *this; } /** * @brief Rolloff factor * @see @fn_al_keyword{GetSourcef} with @def_al{ROLLOFF_FACTOR} */ Float rolloffFactor() const { Float factor; alGetSourcef(_id, AL_ROLLOFF_FACTOR, &factor); return factor; } /** * @brief Set rolloff factor * @return Reference to self (for method chaining) * * Default is @cpp 1.0f @ce. * @see @ref setReferenceDistance(), @fn_al_keyword{Sourcef} with * @def_al{ROLLOFF_FACTOR} */ Source& setRolloffFactor(Float factor) { alSourcef(_id, AL_ROLLOFF_FACTOR, factor); return *this; } /** @overload * @see @fn_al_keyword{Sourcei} with @def_al{ROLLOFF_FACTOR} */ Source& setRolloffFactor(Int factor) { alSourcei(_id, AL_ROLLOFF_FACTOR, factor); return *this; } /** * @brief Maximal distance to clamp to * @see @ref setRolloffFactor(), @fn_al_keyword{GetSourcef} with * @def_al{MAX_DISTANCE} */ Float maxDistance() const { Float distance; alGetSourcef(_id, AL_MAX_DISTANCE, &distance); return distance; } /** * @brief Set max distance * @return Reference to self (for method chaining) * * Default is max representable value. * @see @fn_al_keyword{Sourcef} with @def_al{MAX_DISTANCE} */ Source& setMaxDistance(Float distance) { alSourcef(_id, AL_MAX_DISTANCE, distance); return *this; } /** @overload * @see @fn_al_keyword{Sourcei} with @def_al{MAX_DISTANCE} */ Source& setMaxDistance(Int distance) { alSourcei(_id, AL_MAX_DISTANCE, distance); return *this; } /** * @brief Direction * @see @fn_al_keyword{GetSourcefv} with @def_al{DIRECTION} */ Vector3 direction() const { Vector3 direction; alGetSourcefv(_id, AL_DIRECTION, direction.data()); return direction; } /** * @brief Set direction * @return Reference to self (for method chaining) * * Default is @cpp {0.0f, 0.0f, 0.0f} @ce, which means that the source * is not directional. * @see @ref setInnerConeAngle(), @ref setOuterConeAngle(), * @ref setRelative(), @fn_al_keyword{Sourcefv} with * @def_al{DIRECTION} */ Source& setDirection(const Vector3& direction) { alSourcefv(_id, AL_DIRECTION, direction.data()); return *this; } /** @overload * @see @fn_al_keyword{Sourceiv} with @def_al{DIRECTION} */ Source& setDirection(const Vector3i& direction) { alSourceiv(_id, AL_DIRECTION, direction.data()); return *this; } /** * @brief Inner cone angle * @see @fn_al_keyword{GetSourcef} with @def_al{CONE_INNER_ANGLE} */ Deg innerConeAngle() const { Float angle; alGetSourcef(_id, AL_CONE_INNER_ANGLE, &angle); return Deg(angle); } /** * @brief Set inner cone angle * @return Reference to self (for method chaining) * * Has effect only if the source is directional. Default is * @cpp 360.0_degf @ce. * @see @ref setOuterConeAngle(), @ref setDirection(), * @fn_al_keyword{Sourcef} with @def_al{CONE_INNER_ANGLE} */ Source& setInnerConeAngle(Deg angle) { alSourcef(_id, AL_CONE_INNER_ANGLE, Float(angle)); return *this; } /** * @brief Outer cone angle * @see @fn_al_keyword{GetSourcef} with @def_al{CONE_OUTER_ANGLE} */ Deg outerConeAngle() const { Float angle; alGetSourcef(_id, AL_CONE_OUTER_ANGLE, &angle); return Deg(angle); } /** * @brief Set outer cone angle * @return Reference to self (for method chaining) * * Has effect only if the source is directional. Default is * @cpp 360.0_degf @ce. * @see @ref setInnerConeAngle(), @ref setDirection(), * @ref setOuterConeGain() @fn_al_keyword{Sourcef} with * @def_al{CONE_OUTER_ANGLE} */ Source& setOuterConeAngle(Deg angle) { alSourcef(_id, AL_CONE_OUTER_ANGLE, Float(angle)); return *this; } /** * @brief Outer cone gain * @see @fn_al_keyword{GetSourcef} with @def_al{CONE_OUTER_GAIN} */ Float outerConeGain() const { Float gain; alGetSourcef(_id, AL_CONE_OUTER_GAIN, &gain); return gain; } /** * @brief Set outer cone gain multiplier * @return Reference to self (for method chaining) * * The factor with which the gain is multiplied outside the outer cone. * Default is @cpp 0.0f @ce. * @see @ref setGain(), @ref setOuterConeAngle(), * @fn_al_keyword{Sourcef} with @def_al{CONE_OUTER_GAIN} */ Source& setOuterConeGain(Float multiplier) { alSourcef(_id, AL_CONE_OUTER_GAIN, multiplier); return *this; } /** * @brief Pitch * @see @fn_al_keyword{GetSourcef} with @def_al{PITCH} */ Float pitch() const { Float pitch; alGetSourcef(_id, AL_PITCH, &pitch); return pitch; } /** * @brief Set pitch * @return Reference to self (for method chaining) * * Default is @cpp 1.0f @ce. * @see @fn_al_keyword{Sourcef} with @def_al{PITCH} */ Source& setPitch(Float pitch) { alSourcef(_id, AL_PITCH, pitch); return *this; } /*@}*/ /** @{ @name Buffer management */ /** * @brief Source type * * @see @ref type() */ enum class Type: ALint { Undetermined = AL_UNDETERMINED, /**< Undetermined (default) */ Static = AL_STATIC, /**< Static source */ Streaming = AL_STREAMING /**< Streaming source */ }; /** * @brief Source type * * @see @ref setBuffer(), @fn_al_keyword{GetSourcei} with * @def_al{SOURCE_TYPE} */ Type type() const; /** * @brief Attach buffer * @param buffer Buffer to attach or @cpp nullptr @ce * @return Reference to self (for method chaining) * * If an buffer is attached, changes source type to @ref Type::Static, * if detached, changes source type to @ref Type::Undetermined. The * buffer must be already filled with data. * @see @ref type(), @fn_al_keyword{Sourcei} with @def_al{BUFFER} */ Source& setBuffer(Buffer* buffer); /*@}*/ /** @{ @name State management */ /** * @brief Source state * * @see @ref state(), @ref play(), @ref pause(), @ref stop(), * @ref rewind() */ enum class State: ALint { Initial = AL_INITIAL, /**< Initial state (default) */ Playing = AL_PLAYING, /**< The source is playing */ Paused = AL_PAUSED, /**< The source is paused */ Stopped = AL_STOPPED /**< The source is stopped */ }; /** * @brief Play more sources at once * * The operation is guaranteed to be done for all sources at the same * time. * @see @ref play(), @ref pause(std::initializer_list>), * @ref stop(std::initializer_list>), * @ref rewind(std::initializer_list>), * @fn_al_keyword{SourcePlayv} */ static void play(std::initializer_list> sources); static void play(const std::vector>& sources); /**< @overload */ /** * @brief Pause more sources at once * * The operation is guaranteed to be done for all sources at the same * time. * @see @ref pause(), @ref play(std::initializer_list>), * @ref stop(std::initializer_list>), * @ref rewind(std::initializer_list>), * @fn_al_keyword{SourcePausev} */ static void pause(std::initializer_list> sources); static void pause(const std::vector>& sources); /**< @overload */ /** * @brief Stop more sources at once * * The operation is guaranteed to be done for all sources at the same * time. * @see @ref stop(), @ref play(std::initializer_list>), * @ref pause(std::initializer_list>), * @ref rewind(std::initializer_list>), * @fn_al_keyword{SourceStopv} */ static void stop(std::initializer_list> sources); static void stop(const std::vector>& sources); /**< @overload */ /** * @brief Rewind more sources at once * * The operation is guaranteed to be done for all sources at the same * time. * @see @ref rewind(), @ref play(std::initializer_list>), * @ref pause(std::initializer_list>), * @ref stop(std::initializer_list>), * @fn_al_keyword{SourceRewindv} */ static void rewind(std::initializer_list> sources); static void rewind(const std::vector>& sources); /**< @overload */ /** * @brief State * * @see @ref play(), @ref pause(), @ref stop(), @ref rewind(), * @fn_al_keyword{GetSourcei} with @def_al{SOURCE_STATE} */ State state() const; /** * @brief Play * * @see @ref play(std::initializer_list>), * @ref state(), @ref pause(), @ref stop(), @ref rewind(), * @fn_al_keyword{SourcePlay} */ void play() { alSourcePlay(_id); } /** * @brief Pause * * @see @ref pause(std::initializer_list>), * @ref state(), @ref play(), @ref stop(), @ref rewind(), * @fn_al_keyword{SourcePause} */ void pause() { alSourcePause(_id); } /** * @brief Stop * * @see @ref stop(std::initializer_list>), * @ref state(), @ref play(), @ref pause(), @ref rewind(), * @fn_al_keyword{SourceStop} */ void stop() { alSourceStop(_id); } /** * @brief Rewind * * @see @ref rewind(std::initializer_list>), * @ref state(), @ref play(), @ref pause(), @ref stop(), * @fn_al_keyword{SourceRewind} */ void rewind() { alSourceRewind(_id); } /** * @brief Whether the source is looping * * @see @fn_al_keyword{GetSourcei} with @def_al{LOOPING} */ bool isLooping() const; /** * @brief Set source looping * @return Reference to self (for method chaining) * * Default is @cpp false @ce. * @see @fn_al_keyword{Sourcei} with @def_al{LOOPING} */ Source& setLooping(bool loop) { alSourcei(_id, AL_LOOPING, loop); return *this; } /** * @brief Offset in seconds * * @see @ref offsetInBytes(), @ref offsetInSamples(), * @fn_al_keyword{GetSourcef} with @def_al{SEC_OFFSET} */ Float offsetInSeconds() const; /** * @brief Set offset in seconds * @return Reference to self (for method chaining) * * @see @ref setOffsetInBytes(), @ref setOffsetInSamples(), * @fn_al_keyword{Sourcef} with @def_al{SEC_OFFSET} */ Source& setOffsetInSeconds(Float offset) { alSourcef(_id, AL_SEC_OFFSET, offset); return *this; } /** * @brief Offset in bytes * * @see @ref offsetInSeconds(), @ref offsetInSamples(), * @fn_al_keyword{GetSourcei} with @def_al{BYTE_OFFSET} */ Int offsetInBytes() const; /** * @brief Set offset in bytes * @return Reference to self (for method chaining) * * @see @ref setOffsetInSeconds(), @ref setOffsetInSamples(), * @fn_al_keyword{Sourcei} with @def_al{SEC_OFFSET} */ Source& setOffsetInBytes(Int offset) { alSourcei(_id, AL_BYTE_OFFSET, offset); return *this; } /** * @brief Offset in samples * * @see @ref offsetInSeconds(), @ref offsetInBytes(), * @fn_al_keyword{GetSourcei} with @def_al{SAMPLE_OFFSET} */ Int offsetInSamples() const; /** * @brief Set offset in samples * @return Reference to self (for method chaining) * * @see @ref setOffsetInSeconds(), @ref setOffsetInBytes(), * @fn_al_keyword{Sourcei} with @def_al{SEC_OFFSET} */ Source& setOffsetInSamples(Int offset) { alSourcei(_id, AL_SAMPLE_OFFSET, offset); return *this; } /*@}*/ private: ALuint _id; }; /** @debugoperatorclassenum{Magnum::Audio::Source,Magnum::Audio::Source::State} */ MAGNUM_AUDIO_EXPORT Debug& operator<<(Debug& debug, Source::State value); inline Source::Source(Source&& other): _id(other._id) { other._id = 0; } inline Source& Source::operator=(Source&& other) { using std::swap; swap(_id, other._id); return *this; } inline auto Source::state() const -> State { ALint state; alGetSourcei(_id, AL_SOURCE_STATE, &state); return State(state); } inline bool Source::isLooping() const { ALint looping; alGetSourcei(_id, AL_LOOPING, &looping); return looping; } inline Float Source::offsetInSeconds() const { Float offset; alGetSourcef(_id, AL_SEC_OFFSET, &offset); return offset; } inline Int Source::offsetInBytes() const { Int offset; alGetSourcei(_id, AL_BYTE_OFFSET, &offset); return offset; } inline Int Source::offsetInSamples() const { Int offset; alGetSourcei(_id, AL_SAMPLE_OFFSET, &offset); return offset; } }} #endif