aboutsummaryrefslogtreecommitdiff
path: root/src/aomusicplayer.cpp
diff options
context:
space:
mode:
authorSalanto <62221668+Salanto@users.noreply.github.com>2024-05-18 03:33:43 +0200
committerGitHub <noreply@github.com>2024-05-18 03:33:43 +0200
commit51338e30e046f1a62c84049b34577f0f3b5db6e3 (patch)
tree283c95f8f776ed852d03a816388695afd65c0f73 /src/aomusicplayer.cpp
parent469e293a1885e49f1d5994d41ac7cfc9f4666952 (diff)
parentf9c7205210fb9f768f583065971ece09e6b5eb03 (diff)
Merge pull request #957 from AttorneyOnline/kaleidoscope
The Kaleidoscope - Hell, it's about time.
Diffstat (limited to 'src/aomusicplayer.cpp')
-rw-r--r--src/aomusicplayer.cpp281
1 files changed, 165 insertions, 116 deletions
diff --git a/src/aomusicplayer.cpp b/src/aomusicplayer.cpp
index 3f7c43ee..4265a463 100644
--- a/src/aomusicplayer.cpp
+++ b/src/aomusicplayer.cpp
@@ -1,77 +1,99 @@
#include "aomusicplayer.h"
-#include "options.h"
-#include "bass.h"
#include "file_functions.h"
+#include "options.h"
-AOMusicPlayer::AOMusicPlayer(QWidget *parent, AOApplication *p_ao_app)
-{
- m_parent = parent;
- ao_app = p_ao_app;
-}
+#include <bass.h>
+
+#include <QDebug>
+#include <QFuture>
+#include <QWidget>
+
+AOMusicPlayer::AOMusicPlayer(AOApplication *ao_app)
+ : ao_app(ao_app)
+{}
AOMusicPlayer::~AOMusicPlayer()
{
- for (int n_stream = 0; n_stream < m_channelmax; ++n_stream) {
+ for (int n_stream = 0; n_stream < STREAM_COUNT; ++n_stream)
+ {
BASS_ChannelStop(m_stream_list[n_stream]);
}
}
-QString AOMusicPlayer::play(QString p_song, int channel, bool loop,
- int effect_flags)
+QString AOMusicPlayer::playStream(QString song, int streamId, bool loopEnabled, int effectFlags)
{
- channel = channel % m_channelmax;
- if (channel < 0) // wtf?
+ if (!ensureValidStreamId(streamId))
+ {
return "[ERROR] Invalid Channel";
- unsigned int flags = BASS_STREAM_PRESCAN | BASS_STREAM_AUTOFREE |
- BASS_UNICODE | BASS_ASYNCFILE;
- unsigned int streaming_flags = BASS_STREAM_AUTOFREE;
- if (loop) {
+ }
+
+ quint32 flags = BASS_STREAM_AUTOFREE;
+ if (loopEnabled)
+ {
flags |= BASS_SAMPLE_LOOP;
- streaming_flags |= BASS_SAMPLE_LOOP;
}
- QString f_path = p_song;
- DWORD newstream;
- if (f_path.startsWith("http")) {
- if (!Options::getInstance().streamingEnabled()) {
- BASS_ChannelStop(m_stream_list[channel]);
- return QObject::tr("[MISSING] Streaming disabled.");
+ QString f_path = song;
+ DWORD newstream;
+ if (f_path.startsWith("http"))
+ {
+ if (!Options::getInstance().streamingEnabled())
+ {
+ BASS_ChannelStop(m_stream_list[streamId]);
+ return QObject::tr("[MISSING] Streaming disabled.");
}
QUrl l_url = QUrl(f_path);
- newstream = BASS_StreamCreateURL(l_url.toEncoded().toStdString().c_str(), 0, streaming_flags, nullptr, 0);
+ newstream = BASS_StreamCreateURL(l_url.toEncoded().toStdString().c_str(), 0, flags, nullptr, 0);
}
- else {
- f_path = ao_app->get_real_path(ao_app->get_music_path(p_song));
- if (f_path.endsWith(".mo3") || f_path.endsWith(".xm") || f_path.endsWith(".mod") || f_path.endsWith(".s3m") || f_path.endsWith(".it") || f_path.endsWith(".mtm") || f_path.endsWith(".umx") )
- newstream = BASS_MusicLoad(FALSE,f_path.utf16(), 0, 0, flags, 1);
+ else
+ {
+ flags |= BASS_STREAM_PRESCAN | BASS_UNICODE | BASS_ASYNCFILE;
+
+ f_path = ao_app->get_real_path(ao_app->get_music_path(song));
+
+ QString extension = f_path.split('.').last();
+ static const QStringList VALID_EXTENSION_LIST{"mo3", "xm", "mod", "s3m", "it", "mtm", "umx"};
+ if (VALID_EXTENSION_LIST.contains(extension, Qt::CaseInsensitive))
+ {
+ newstream = BASS_MusicLoad(FALSE, f_path.utf16(), 0, 0, flags, 1);
+ }
else
+ {
newstream = BASS_StreamCreateFile(FALSE, f_path.utf16(), 0, 0, flags);
+ }
}
- int error_code = BASS_ErrorGetCode();
-
+ int error = BASS_ErrorGetCode();
if (Options::getInstance().audioOutputDevice() != "default")
- BASS_ChannelSetDevice(m_stream_list[channel], BASS_GetDevice());
+ {
+ BASS_ChannelSetDevice(m_stream_list[streamId], BASS_GetDevice());
+ }
- QString d_path = f_path + ".txt";
+ m_loop_start[streamId] = 0;
+ m_loop_end[streamId] = 0;
- loop_start[channel] = 0;
- loop_end[channel] = 0;
- if (loop && file_exists(d_path)) // Contains loop/etc. information file
+ QString d_path = f_path + ".txt";
+ if (loopEnabled && file_exists(d_path)) // Contains loop/etc. information file
{
QStringList lines = ao_app->read_file(d_path).split("\n");
bool seconds_mode = false;
- foreach (QString line, lines) {
+ foreach (QString line, lines)
+ {
QStringList args = line.split("=");
if (args.size() < 2)
+ {
continue;
+ }
QString arg = args[0].trimmed();
- if (arg == "seconds") {
- if (args[1].trimmed() == "true") {
+ if (arg == "seconds")
+ {
+ if (args[1].trimmed() == "true")
+ {
seconds_mode = true; // Use new epic behavior
continue;
}
+
continue;
}
@@ -86,117 +108,133 @@ QString AOMusicPlayer::play(QString p_song, int channel, bool loop,
// Calculate the bytes for loop_start/loop_end to use with the sync proc
QWORD bytes;
- if (seconds_mode) {
- bytes =
- BASS_ChannelSeconds2Bytes(newstream, args[1].trimmed().toDouble());
+ if (seconds_mode)
+ {
+ bytes = BASS_ChannelSeconds2Bytes(newstream, args[1].trimmed().toDouble());
}
- else {
- bytes = static_cast<QWORD>(args[1].trimmed().toUInt() * sample_size *
- num_channels);
+ else
+ {
+ bytes = static_cast<QWORD>(args[1].trimmed().toUInt() * sample_size * num_channels);
}
if (arg == "loop_start")
- loop_start[channel] = bytes;
+ {
+ m_loop_start[streamId] = bytes;
+ }
else if (arg == "loop_length")
- loop_end[channel] = loop_start[channel] + bytes;
+ {
+ m_loop_end[streamId] = m_loop_start[streamId] + bytes;
+ }
else if (arg == "loop_end")
- loop_end[channel] = bytes;
+ {
+ m_loop_end[streamId] = bytes;
+ }
}
- qDebug() << "Found data file for song" << p_song << "length"
- << BASS_ChannelGetLength(newstream, BASS_POS_BYTE) << "loop start"
- << loop_start[channel] << "loop end" << loop_end[channel];
+ qDebug() << "Found data file for song" << song << "length" << BASS_ChannelGetLength(newstream, BASS_POS_BYTE) << "loop start" << m_loop_start[streamId] << "loop end" << m_loop_end[streamId];
}
- if (BASS_ChannelIsActive(m_stream_list[channel]) == BASS_ACTIVE_PLAYING) {
- DWORD oldstream = m_stream_list[channel];
+ if (BASS_ChannelIsActive(m_stream_list[streamId]) == BASS_ACTIVE_PLAYING)
+ {
+ DWORD oldstream = m_stream_list[streamId];
- if (effect_flags & SYNC_POS) {
+ if (effectFlags & SYNC_POS)
+ {
BASS_ChannelLock(oldstream, true);
// Sync it with the new sample
- BASS_ChannelSetPosition(newstream,
- BASS_ChannelGetPosition(oldstream, BASS_POS_BYTE),
- BASS_POS_BYTE);
+ BASS_ChannelSetPosition(newstream, BASS_ChannelGetPosition(oldstream, BASS_POS_BYTE), BASS_POS_BYTE);
BASS_ChannelLock(oldstream, false);
}
- if ((effect_flags & FADE_OUT) && m_volume[channel] > 0) {
+ if ((effectFlags & FADE_OUT) && m_volume[streamId] > 0)
+ {
// Fade out the other sample and stop it (due to -1)
- BASS_ChannelSlideAttribute(oldstream, BASS_ATTRIB_VOL | BASS_SLIDE_LOG,
- -1, 4000);
+ BASS_ChannelSlideAttribute(oldstream, BASS_ATTRIB_VOL | BASS_SLIDE_LOG, -1, 4000);
}
else
- BASS_ChannelStop(
- oldstream); // Stop the sample since we don't need it anymore
+ {
+ BASS_ChannelStop(oldstream); // Stop the sample since we don't need it anymore
+ }
}
else
- BASS_ChannelStop(m_stream_list[channel]);
+ {
+ BASS_ChannelStop(m_stream_list[streamId]);
+ }
- m_stream_list[channel] = newstream;
+ m_stream_list[streamId] = newstream;
BASS_ChannelPlay(newstream, false);
- if (effect_flags & FADE_IN) {
+ if (effectFlags & FADE_IN)
+ {
// Fade in our sample
BASS_ChannelSetAttribute(newstream, BASS_ATTRIB_VOL, 0);
- BASS_ChannelSlideAttribute(newstream, BASS_ATTRIB_VOL,
- static_cast<float>(m_volume[channel] / 100.0f),
- 1000);
+ BASS_ChannelSlideAttribute(newstream, BASS_ATTRIB_VOL, static_cast<float>(m_volume[streamId] / 100.0f), 1000);
}
else
- this->set_volume(m_volume[channel], channel);
+ {
+ this->setStreamVolume(m_volume[streamId], streamId);
+ }
- BASS_ChannelSetSync(newstream, BASS_SYNC_DEV_FAIL, 0,
- ao_app->BASSreset, 0);
+ BASS_ChannelSetSync(newstream, BASS_SYNC_DEV_FAIL, 0, ao_app->BASSreset, 0);
- this->set_looping(loop, channel); // Have to do this here due to any
- // crossfading-related changes, etc.
+ this->setStreamLooping(loopEnabled, streamId); // Have to do this here due to any
+ // crossfading-related changes, etc.
- bool is_stop = (p_song == "~stop.mp3");
- QString p_song_clear = QUrl(p_song).fileName();
+ bool is_stop = (song == "~stop.mp3");
+ QString p_song_clear = QUrl(song).fileName();
p_song_clear = p_song_clear.left(p_song_clear.lastIndexOf('.'));
- if (is_stop && channel == 0) { // don't send text on channels besides 0
+ if (is_stop && streamId == 0)
+ { // don't send text on channels besides 0
return QObject::tr("None");
}
- if (error_code == BASS_ERROR_HANDLE) { // Cheap hack to see if file missing
+ if (error == BASS_ERROR_HANDLE)
+ { // Cheap hack to see if file missing
return QObject::tr("[MISSING] %1").arg(p_song_clear);
}
- if (p_song.startsWith("http") && channel == 0) {
+ if (song.startsWith("http") && streamId == 0)
+ {
return QObject::tr("[STREAM] %1").arg(p_song_clear);
}
- if (channel == 0)
+ if (streamId == 0)
+ {
return p_song_clear;
+ }
return "";
}
-void AOMusicPlayer::stop(int channel)
-{
- BASS_ChannelStop(m_stream_list[channel]);
-}
-
-void AOMusicPlayer::set_muted(bool toggle)
+void AOMusicPlayer::setMuted(bool enabled)
{
- m_muted = toggle;
+ m_muted = enabled;
// Update all volume based on the mute setting
- for (int n_stream = 0; n_stream < m_channelmax; ++n_stream) {
- set_volume(m_volume[n_stream], n_stream);
+ for (int n_stream = 0; n_stream < STREAM_COUNT; ++n_stream)
+ {
+ setStreamVolume(m_volume[n_stream], n_stream);
}
}
-void AOMusicPlayer::set_volume(int p_value, int channel)
+void AOMusicPlayer::setStreamVolume(int value, int streamId)
{
- m_volume[channel] = p_value;
+ if (!ensureValidStreamId(streamId))
+ {
+ qWarning().noquote() << QObject::tr("Invalid stream ID '%2'").arg(streamId);
+ return;
+ }
+
+ m_volume[streamId] = value;
// If muted, volume will always be 0
- float volume = (m_volume[channel] / 100.0f) * !m_muted;
- if (channel < 0) {
- for (int n_stream = 0; n_stream < m_channelmax; ++n_stream) {
- BASS_ChannelSetAttribute(m_stream_list[n_stream], BASS_ATTRIB_VOL,
- volume);
+ float volume = (m_volume[streamId] / 100.0f) * !m_muted;
+ if (streamId < 0)
+ {
+ for (int n_stream = 0; n_stream < STREAM_COUNT; ++n_stream)
+ {
+ BASS_ChannelSetAttribute(m_stream_list[n_stream], BASS_ATTRIB_VOL, volume);
}
}
- else {
- BASS_ChannelSetAttribute(m_stream_list[channel], BASS_ATTRIB_VOL, volume);
+ else
+ {
+ BASS_ChannelSetAttribute(m_stream_list[streamId], BASS_ATTRIB_VOL, volume);
}
}
@@ -210,37 +248,48 @@ void CALLBACK loopProc(HSYNC handle, DWORD channel, DWORD data, void *user)
BASS_ChannelLock(channel, false);
}
-void AOMusicPlayer::set_looping(bool loop_song, int channel)
+void AOMusicPlayer::setStreamLooping(bool enabled, int streamId)
{
- if (!loop_song) {
- if (BASS_ChannelFlags(m_stream_list[channel], 0, 0) & BASS_SAMPLE_LOOP)
- BASS_ChannelFlags(m_stream_list[channel], 0,
+ if (!ensureValidStreamId(streamId))
+ {
+ qWarning().noquote() << QObject::tr("Invalid stream ID '%2'").arg(streamId);
+ return;
+ }
+
+ if (!enabled)
+ {
+ if (BASS_ChannelFlags(m_stream_list[streamId], 0, 0) & BASS_SAMPLE_LOOP)
+ {
+ BASS_ChannelFlags(m_stream_list[streamId], 0,
BASS_SAMPLE_LOOP); // remove the LOOP flag
- BASS_ChannelRemoveSync(m_stream_list[channel], loop_sync[channel]);
- loop_sync[channel] = 0;
+ }
+ BASS_ChannelRemoveSync(m_stream_list[streamId], m_loop_sync[streamId]);
+ m_loop_sync[streamId] = 0;
return;
}
- BASS_ChannelFlags(m_stream_list[channel], BASS_SAMPLE_LOOP,
+ BASS_ChannelFlags(m_stream_list[streamId], BASS_SAMPLE_LOOP,
BASS_SAMPLE_LOOP); // set the LOOP flag
- if (loop_sync[channel] != 0) {
- BASS_ChannelRemoveSync(m_stream_list[channel],
- loop_sync[channel]); // remove the sync
- loop_sync[channel] = 0;
+ if (m_loop_sync[streamId] != 0)
+ {
+ BASS_ChannelRemoveSync(m_stream_list[streamId],
+ m_loop_sync[streamId]); // remove the sync
+ m_loop_sync[streamId] = 0;
}
- if (loop_start[channel] < loop_end[channel])
+ if (m_loop_start[streamId] < m_loop_end[streamId])
{
- //Loop when the endpoint is reached.
- loop_sync[channel] = BASS_ChannelSetSync(
- m_stream_list[channel], BASS_SYNC_POS | BASS_SYNC_MIXTIME,
- loop_end[channel], loopProc, &loop_start[channel]);
+ // Loop when the endpoint is reached.
+ m_loop_sync[streamId] = BASS_ChannelSetSync(m_stream_list[streamId], BASS_SYNC_POS | BASS_SYNC_MIXTIME, m_loop_end[streamId], loopProc, &m_loop_start[streamId]);
}
else
{
- //Loop when the end of the file is reached.
- loop_sync[channel] = BASS_ChannelSetSync(
- m_stream_list[channel], BASS_SYNC_END | BASS_SYNC_MIXTIME,
- 0, loopProc, &loop_start[channel]);
+ // Loop when the end of the file is reached.
+ m_loop_sync[streamId] = BASS_ChannelSetSync(m_stream_list[streamId], BASS_SYNC_END | BASS_SYNC_MIXTIME, 0, loopProc, &m_loop_start[streamId]);
}
}
+
+bool AOMusicPlayer::ensureValidStreamId(int streamId)
+{
+ return (streamId >= 0 && streamId < STREAM_COUNT);
+}