2025-10-27 05:15:47 -05:00
|
|
|
/**
|
|
|
|
|
* Furnace Tracker - multi-system chiptune tracker
|
|
|
|
|
* Copyright (C) 2021-2025 tildearrow and contributors
|
|
|
|
|
*
|
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "gui.h"
|
|
|
|
|
#include <fmt/printf.h>
|
|
|
|
|
#include "imgui.h"
|
|
|
|
|
#include "IconsFontAwesome4.h"
|
2025-10-31 03:42:43 -05:00
|
|
|
#include "misc/cpp/imgui_stdlib.h"
|
2025-10-27 05:15:47 -05:00
|
|
|
|
|
|
|
|
void FurnaceGUI::drawRefPlayer() {
|
2025-10-30 01:30:48 -05:00
|
|
|
DivFilePlayer* fp=e->getFilePlayer();
|
2025-10-27 05:15:47 -05:00
|
|
|
if (nextWindow==GUI_WINDOW_REF_PLAYER) {
|
|
|
|
|
refPlayerOpen=true;
|
|
|
|
|
ImGui::SetNextWindowFocus();
|
|
|
|
|
nextWindow=GUI_WINDOW_NOTHING;
|
|
|
|
|
}
|
2025-10-30 01:30:48 -05:00
|
|
|
fp->setActive(refPlayerOpen);
|
2025-10-27 05:15:47 -05:00
|
|
|
if (!refPlayerOpen) return;
|
|
|
|
|
|
2025-10-30 01:30:48 -05:00
|
|
|
if (ImGui::Begin("Music Player",&refPlayerOpen,globalWinFlags,_("Music Player"))) {
|
2025-10-27 19:34:21 -05:00
|
|
|
bool playPosNegative=false;
|
|
|
|
|
ssize_t playPos=fp->getPos();
|
|
|
|
|
if (playPos<0) {
|
|
|
|
|
playPos=-playPos;
|
|
|
|
|
playPosNegative=true;
|
|
|
|
|
}
|
2025-10-27 05:15:47 -05:00
|
|
|
size_t minPos=0;
|
|
|
|
|
size_t maxPos=fp->getFileInfo().frames;
|
|
|
|
|
int fileRate=fp->getFileInfo().samplerate;
|
|
|
|
|
if (fileRate<1) fileRate=1;
|
|
|
|
|
int posHours=(playPos/fileRate)/3600;
|
|
|
|
|
int posMinutes=((playPos/fileRate)/60)%60;
|
|
|
|
|
int posSeconds=(playPos/fileRate)%60;
|
|
|
|
|
int posMillis=(1000*(playPos%fileRate))/fileRate;
|
2025-10-30 04:07:27 -05:00
|
|
|
if (fp->isLoaded()) {
|
|
|
|
|
if (playPosNegative) {
|
|
|
|
|
ImGui::Text("-%d:%02d:%02d.%03d",posHours,posMinutes,posSeconds,posMillis);
|
|
|
|
|
} else {
|
|
|
|
|
ImGui::Text("%d:%02d:%02d.%03d",posHours,posMinutes,posSeconds,posMillis);
|
|
|
|
|
}
|
2025-10-27 19:34:21 -05:00
|
|
|
} else {
|
2025-10-30 04:07:27 -05:00
|
|
|
ImGui::TextUnformatted(_("no file loaded"));
|
2025-10-27 19:34:21 -05:00
|
|
|
}
|
2025-10-27 05:15:47 -05:00
|
|
|
|
|
|
|
|
ImGui::SetNextItemWidth(ImGui::GetContentRegionAvail().x);
|
|
|
|
|
if (ImGui::SliderScalar("##Position",ImGuiDataType_U64,&playPos,&minPos,&maxPos,"")) {
|
|
|
|
|
fp->setPos(playPos);
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-30 04:07:27 -05:00
|
|
|
if (ImGui::Button(ICON_FA_FOLDER_OPEN "##Open")) {
|
2025-10-27 05:15:47 -05:00
|
|
|
openFileDialog(GUI_FILE_MUSIC_OPEN);
|
|
|
|
|
}
|
2025-10-30 04:07:27 -05:00
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
|
|
|
|
e->synchronizedSoft([this,fp]() {
|
|
|
|
|
if (!fp->closeFile()) {
|
|
|
|
|
showError(_("you haven't loaded a file!"));
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
ImGui::SetItemTooltip(_("open file\n(right click to unload current file)"));
|
2025-10-27 05:15:47 -05:00
|
|
|
ImGui::SameLine();
|
2025-10-30 04:07:27 -05:00
|
|
|
if (ImGui::Button(ICON_FA_STEP_BACKWARD)) {
|
|
|
|
|
// handled outside
|
|
|
|
|
}
|
|
|
|
|
if (fp->isPlaying()) {
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Left)) {
|
|
|
|
|
fp->stop();
|
2025-10-30 20:35:14 -05:00
|
|
|
fp->setPosSeconds(e->getFilePlayerCue());
|
2025-10-30 04:07:27 -05:00
|
|
|
}
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Middle)) {
|
|
|
|
|
fp->stop();
|
|
|
|
|
fp->setPos(0);
|
|
|
|
|
}
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
2025-10-30 20:35:14 -05:00
|
|
|
fp->setPosSeconds(e->getFilePlayerCue());
|
2025-10-30 04:07:27 -05:00
|
|
|
}
|
|
|
|
|
ImGui::SetItemTooltip(
|
|
|
|
|
_("left click: go to cue position\n"
|
|
|
|
|
"middle click: go to beginning\n"
|
|
|
|
|
"right click: go to cue position (but don't stop)")
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Left)) {
|
|
|
|
|
// try setting cue pos
|
2025-10-30 20:35:14 -05:00
|
|
|
TimeMicros curPos=fp->getPosSeconds();
|
2025-10-30 18:44:59 -05:00
|
|
|
TimeMicros rowTS=e->curSubSong->ts.getTimes(curOrder,0);
|
2025-10-30 04:07:27 -05:00
|
|
|
if (rowTS.seconds==-1) {
|
2025-10-30 04:53:17 -05:00
|
|
|
showError(_("the first row of this order isn't going to play."));
|
2025-10-30 04:07:27 -05:00
|
|
|
} else {
|
2025-10-30 20:35:14 -05:00
|
|
|
e->setFilePlayerCue(curPos-rowTS);
|
2025-10-30 04:07:27 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Middle)) {
|
|
|
|
|
fp->setPos(0);
|
|
|
|
|
}
|
|
|
|
|
if (ImGui::BeginPopupContextItem("Edit Cue Position",ImGuiPopupFlags_MouseButtonRight)) {
|
2025-10-30 04:53:17 -05:00
|
|
|
ImGui::TextUnformatted(_("Set cue position at first order:"));
|
2025-10-30 20:35:14 -05:00
|
|
|
TimeMicros cueTime=e->getFilePlayerCue();
|
2025-10-30 04:35:44 -05:00
|
|
|
bool altered=false;
|
2025-10-31 03:42:43 -05:00
|
|
|
fpCueInput=cueTime.toString(-1,TA_TIME_FORMAT_AUTO);
|
|
|
|
|
pushWarningColor(false,fpCueInputFailed);
|
|
|
|
|
if (ImGui::InputText("##CuePos",&fpCueInput)) {
|
|
|
|
|
try {
|
|
|
|
|
cueTime=TimeMicros::fromString(fpCueInput);
|
|
|
|
|
altered=true;
|
|
|
|
|
fpCueInputFailed=false;
|
|
|
|
|
} catch (std::invalid_argument& e) {
|
|
|
|
|
fpCueInputFailed=true;
|
|
|
|
|
fpCueInputFailReason=e.what();
|
|
|
|
|
}
|
2025-10-30 04:35:44 -05:00
|
|
|
}
|
2025-10-31 03:42:43 -05:00
|
|
|
if (!ImGui::IsItemActive()) {
|
|
|
|
|
fpCueInputFailed=false;
|
2025-10-30 04:35:44 -05:00
|
|
|
}
|
2025-10-31 03:42:43 -05:00
|
|
|
if (ImGui::IsItemHovered() && fpCueInputFailed) {
|
|
|
|
|
ImGui::SetTooltip("%s",fpCueInputFailReason.c_str());
|
|
|
|
|
}
|
|
|
|
|
popWarningColor();
|
2025-10-30 04:35:44 -05:00
|
|
|
if (altered) {
|
2025-10-30 20:35:14 -05:00
|
|
|
e->setFilePlayerCue(cueTime);
|
2025-10-30 04:35:44 -05:00
|
|
|
}
|
2025-10-30 04:53:17 -05:00
|
|
|
if (ImGui::Button(_("OK"))) {
|
2025-10-30 04:07:27 -05:00
|
|
|
ImGui::CloseCurrentPopup();
|
|
|
|
|
}
|
|
|
|
|
ImGui::EndPopup();
|
|
|
|
|
}
|
|
|
|
|
ImGui::SetItemTooltip(
|
|
|
|
|
_("left click: set cue position here\n"
|
|
|
|
|
" - current playback time becomes position at first row of current order\n"
|
|
|
|
|
"middle click: go to beginning\n"
|
|
|
|
|
"right click: fine edit cue position")
|
|
|
|
|
);
|
2025-10-27 05:15:47 -05:00
|
|
|
}
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
if (fp->isPlaying()) {
|
|
|
|
|
pushToggleColors(true);
|
|
|
|
|
if (ImGui::Button(ICON_FA_PAUSE "##Pause")) {
|
|
|
|
|
fp->stop();
|
|
|
|
|
}
|
2025-10-30 04:35:44 -05:00
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
|
|
|
|
// try setting cue pos
|
2025-10-30 20:35:14 -05:00
|
|
|
TimeMicros curPos=fp->getPosSeconds();
|
2025-10-30 18:44:59 -05:00
|
|
|
TimeMicros rowTS=e->curSubSong->ts.getTimes(curOrder,0);
|
2025-10-30 04:35:44 -05:00
|
|
|
if (rowTS.seconds==-1) {
|
2025-10-30 04:53:17 -05:00
|
|
|
showError(_("the first row of this order isn't going to play."));
|
2025-10-30 04:35:44 -05:00
|
|
|
} else {
|
2025-10-30 20:35:14 -05:00
|
|
|
e->setFilePlayerCue(curPos-rowTS);
|
2025-10-30 04:35:44 -05:00
|
|
|
fp->stop();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ImGui::SetItemTooltip(_("pause\n(right click to set cue position and pause)"));
|
2025-10-27 05:15:47 -05:00
|
|
|
popToggleColors();
|
|
|
|
|
} else {
|
|
|
|
|
if (ImGui::Button(ICON_FA_PLAY "##Play")) {
|
|
|
|
|
fp->play();
|
|
|
|
|
}
|
2025-10-30 04:07:27 -05:00
|
|
|
ImGui::SetItemTooltip(_("play"));
|
2025-10-27 05:15:47 -05:00
|
|
|
}
|
|
|
|
|
ImGui::SameLine();
|
2025-10-27 19:34:21 -05:00
|
|
|
|
2025-10-30 04:53:17 -05:00
|
|
|
if (ImGui::Button(ICON_FA_STEP_FORWARD "##PlayPos")) {
|
|
|
|
|
// handled outside
|
|
|
|
|
}
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Left) || ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
2025-10-30 18:44:59 -05:00
|
|
|
TimeMicros rowTS;
|
2025-10-30 04:53:17 -05:00
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
|
|
|
|
rowTS=e->curSubSong->ts.getTimes(cursor.order,cursor.y);
|
|
|
|
|
} else {
|
|
|
|
|
rowTS=e->curSubSong->ts.getTimes(curOrder,0);
|
|
|
|
|
}
|
2025-10-30 20:35:14 -05:00
|
|
|
TimeMicros cueTime=e->getFilePlayerCue();
|
2025-10-30 04:53:17 -05:00
|
|
|
if (rowTS.seconds==-1) {
|
|
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
2025-10-30 04:59:14 -05:00
|
|
|
showError(_("the row that the pattern cursor is at isn't going to play. try moving the cursor."));
|
2025-10-30 04:53:17 -05:00
|
|
|
} else {
|
|
|
|
|
showError(_("the first row of this order isn't going to play. try another order."));
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2025-10-30 20:35:14 -05:00
|
|
|
fp->setPosSeconds(rowTS+cueTime);
|
2025-10-30 04:53:17 -05:00
|
|
|
fp->play();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (ImGui::IsItemHovered() && (ImGui::IsMouseReleased(ImGuiMouseButton_Left) || ImGui::IsMouseReleased(ImGuiMouseButton_Right))) {
|
|
|
|
|
fp->stop();
|
|
|
|
|
}
|
|
|
|
|
ImGui::SetItemTooltip(_(
|
|
|
|
|
"hold left click to play from current order\n"
|
|
|
|
|
"hold right click to play from pattern cursor position\n"
|
|
|
|
|
"release mouse button to stop"
|
|
|
|
|
));
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
|
2025-10-29 20:00:08 -05:00
|
|
|
pushToggleColors(filePlayerSync);
|
2025-10-27 19:34:21 -05:00
|
|
|
if (ImGui::Button(_("Sync"))) {
|
2025-10-29 20:00:08 -05:00
|
|
|
filePlayerSync=!filePlayerSync;
|
2025-10-27 19:34:21 -05:00
|
|
|
}
|
2025-10-30 04:07:27 -05:00
|
|
|
ImGui::SetItemTooltip(_("synchronize playback with tracker playback"));
|
2025-10-27 19:34:21 -05:00
|
|
|
popToggleColors();
|
2025-10-29 20:00:08 -05:00
|
|
|
e->setFilePlayerSync(filePlayerSync);
|
2025-10-30 01:30:48 -05:00
|
|
|
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
ImGui::Text(_("Mix:"));
|
2025-10-27 05:15:47 -05:00
|
|
|
|
|
|
|
|
float vol=fp->getVolume();
|
2025-10-29 19:25:08 -05:00
|
|
|
ImGui::SameLine();
|
2025-10-27 19:34:21 -05:00
|
|
|
ImGui::SetNextItemWidth(ImGui::GetContentRegionAvail().x);
|
2025-10-30 01:30:48 -05:00
|
|
|
if (ImGui::SliderFloat("##Volume",&vol,-1.0f,1.0f,_("<-- Tracker / Reference -->"))) {
|
|
|
|
|
if (vol<-1.0f) vol=-1.0f;
|
2025-10-27 05:15:47 -05:00
|
|
|
if (vol>1.0f) vol=1.0f;
|
|
|
|
|
fp->setVolume(vol);
|
|
|
|
|
}
|
2025-10-30 01:30:48 -05:00
|
|
|
if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) {
|
|
|
|
|
fp->setVolume(0.0f);
|
|
|
|
|
}
|
2025-10-30 04:35:44 -05:00
|
|
|
ImGui::SetItemTooltip(_("right click to reset"));
|
2025-10-27 14:24:16 -05:00
|
|
|
|
2025-10-29 19:25:08 -05:00
|
|
|
//ImGui::Text("Memory usage: %" PRIu64 "K",fp->getMemUsage()>>10);
|
2025-10-27 05:15:47 -05:00
|
|
|
}
|
|
|
|
|
if (ImGui::IsWindowFocused(ImGuiFocusedFlags_ChildWindows)) curWindow=GUI_WINDOW_REF_PLAYER;
|
|
|
|
|
ImGui::End();
|
2025-10-30 01:30:48 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
if (!refPlayerOpen) {
|
|
|
|
|
fp->stop();
|
|
|
|
|
e->setFilePlayerSync(false);
|
|
|
|
|
}
|
2025-10-27 05:15:47 -05:00
|
|
|
}
|