ref: 0f13e15d79d27124d3f1dbba1ea17d60a8b1b47d
dir: /src/ft2_sysreqs.c/
#include <stdio.h> // vsnprintf()
#include <stdint.h>
#include <stdbool.h>
#include "ft2_config.h"
#include "ft2_gui.h"
#include "ft2_mouse.h"
#include "ft2_keyboard.h"
#include "ft2_textboxes.h"
#include "ft2_video.h"
#include "ft2_sysreqs.h"
#include "ft2_structs.h"
#include "ft2_events.h"
#define SYSTEM_REQUEST_H 67
#define SYSTEM_REQUEST_Y 249
#define SYSTEM_REQUEST_Y_EXT 91
// globalized
okBoxData_t okBoxData;
void (*loaderMsgBox)(const char *, ...);
int16_t (*loaderSysReq)(int16_t, const char *, const char *, void (*)(void));
// ----------------
#define NUM_SYSREQ_TYPES 6
static char *buttonText[NUM_SYSREQ_TYPES][5] =
{
	// generic dialogs
	{ "OK", "","","","" },
	{ "OK", "Cancel", "","","" },
	{ "Yes", "No", "","","" },
	// custom dialogs
	{ "All", "Song", "Instruments", "Cancel", "" },   // "song clear" dialog
	{ "Read left", "Read right", "Convert", "", "" }, // "stereo sample loader" dialog
	{ "Mono", "Stereo", "Cancel", "","" }             // "audio sampling" dialog
};
static SDL_Keycode shortCut[NUM_SYSREQ_TYPES][5] =
{
	// generic dialogs
	{ SDLK_o, 0,      0,      0,      0 },
	{ SDLK_o, SDLK_c, 0,      0,      0 },
	{ SDLK_y, SDLK_n, 0,      0,      0 },
	// custom dialogs
	{ SDLK_a, SDLK_s, SDLK_i, SDLK_c, 0 }, // "song clear" dialog
	{ SDLK_l, SDLK_r, SDLK_c, 0,      0 }, // "stereo sample loader" dialog 
	{ SDLK_m, SDLK_s, SDLK_c, 0,      0 }, // "audio sampling" dialog
};
typedef struct quitType_t
{
	const char *text;
	uint8_t type;
} quitType_t;
#define QUIT_MESSAGES 11
static quitType_t quitMessage[QUIT_MESSAGES] =
{
	// edited (and removed) some of the original quit texts...
	{ "Do you really want to quit?", 2 },
	{ "Tired already?", 2 },
	{ "Dost thou wish to leave with such hasty abandon?", 2 },
	{ "So, you think you can quit this easily, huh?", 2 },
	{ "Hey, what is the matter? You are not quiting now, are you?", 2 },
	{ "Rome was not built in one day! Quit really?", 2 },
	{ "Did you really press the right key?", 2 },
	{ "Hope ya did some good. Press >OK< to quit.", 1 },
	{ "Quit? Only for a good reason you are allowed to press >OK<.", 1 },
	{ "Are we at the end of a Fasttracker II round?", 2 },
	{ "Hope you're doing the compulsory \"Exit ceremony\" before pressing >OK<.", 1 },
};
void myLoaderMsgBoxThreadSafe(const char *fmt, ...)
{
	char strBuf[512];
	va_list args;
	// format the text string
	va_start(args, fmt);
	vsnprintf(strBuf, sizeof (strBuf), fmt, args);
	va_end(args);
	okBoxThreadSafe(0, "System message", fmt, NULL);
}
void myLoaderMsgBox(const char *fmt, ...)
{
	char strBuf[512];
	va_list args;
	// format the text string
	va_start(args, fmt);
	vsnprintf(strBuf, sizeof (strBuf), fmt, args);
	va_end(args);
	okBox(0, "System message", fmt, NULL);
}
static void drawWindow(uint16_t w)
{
	const uint16_t h = SYSTEM_REQUEST_H;
	const uint16_t x = (SCREEN_W - w) / 2;
	const uint16_t y = ui.extendedPatternEditor ? 91 : SYSTEM_REQUEST_Y;
	// main fill
	fillRect(x + 1, y + 1, w - 2, h - 2, PAL_BUTTONS);
	// outer border
	vLine(x,         y,         h - 1, PAL_BUTTON1);
	hLine(x + 1,     y,         w - 2, PAL_BUTTON1);
	vLine(x + w - 1, y,         h,     PAL_BUTTON2);
	hLine(x,         y + h - 1, w - 1, PAL_BUTTON2);
	// inner border
	vLine(x + 2,     y + 2,     h - 5, PAL_BUTTON2);
	hLine(x + 3,     y + 2,     w - 6, PAL_BUTTON2);
	vLine(x + w - 3, y + 2,     h - 4, PAL_BUTTON1);
	hLine(x + 2,     y + h - 3, w - 4, PAL_BUTTON1);
	// title bottom line
	hLine(x + 3, y + 16, w - 6, PAL_BUTTON2);
	hLine(x + 3, y + 17, w - 6, PAL_BUTTON1);
}
static bool mouseButtonDownLogic(uint8_t mouseButton)
{
	// if already holding left button and clicking right, don't do mouse down handling
	if (mouseButton == SDL_BUTTON_RIGHT && mouse.leftButtonPressed)
	{
		mouse.rightButtonPressed = true;
		return false;
	}
	// if already holding right button and clicking left, don't do mouse down handling
	if (mouseButton == SDL_BUTTON_LEFT && mouse.rightButtonPressed)
	{
		mouse.leftButtonPressed = true;
		return false;
	}
	if (mouseButton == SDL_BUTTON_LEFT)
		mouse.leftButtonPressed = true;
	else if (mouseButton == SDL_BUTTON_RIGHT)
		mouse.rightButtonPressed = true;
	// don't do mouse down testing here if we already are using an object
	if (mouse.lastUsedObjectType != OBJECT_NONE)
		return false;
	// kludge #2
	if (mouse.lastUsedObjectType != OBJECT_PUSHBUTTON && mouse.lastUsedObjectID != OBJECT_ID_NONE)
		return false;
	// kludge #3
	if (!mouse.rightButtonPressed)
		mouse.lastUsedObjectID = OBJECT_ID_NONE;
	return true;
}
static bool mouseButtonUpLogic(uint8_t mouseButton)
{
#if defined __APPLE__ && defined __aarch64__
	armMacGhostMouseCursorFix();
#endif
	if (mouseButton == SDL_BUTTON_LEFT)
		mouse.leftButtonPressed = false;
	else if (mouseButton == SDL_BUTTON_RIGHT)
		mouse.rightButtonPressed = false;
	editor.textCursorBlinkCounter = 0;
	// if we used both mouse button at the same time and released *one*, don't release GUI object
	if ( mouse.leftButtonPressed && !mouse.rightButtonPressed) return false;
	if (!mouse.leftButtonPressed &&  mouse.rightButtonPressed) return false;
	return true;
}
// WARNING: This routine must ONLY be called from the main input/video thread!
// If the checkBoxCallback argument is set, then you get a "Do not show again" checkbox.
int16_t okBox(int16_t type, const char *headline, const char *text, void (*checkBoxCallback)(void))
{
#define PUSHBUTTON_W 80
	SDL_Event inputEvent;
	if (editor.editTextFlag)
		exitTextEditing();
	// revert "delete/rename" mouse modes (disk op.)
	if (mouse.mode != MOUSE_MODE_NORMAL)
		setMouseMode(MOUSE_MODE_NORMAL);
	if (ui.sysReqShown)
		return 0;
	SDL_EventState(SDL_DROPFILE, SDL_DISABLE);
	ui.sysReqShown = true;
	mouseAnimOff();
	int16_t oldLastUsedObjectID = mouse.lastUsedObjectID;
	int16_t oldLastUsedObjectType = mouse.lastUsedObjectType;
	// count number of buttons
	uint16_t numButtons = 0;
	while (buttonText[type][numButtons][0] != '\0' && numButtons < 5)
		numButtons++;
	uint16_t tlen = textWidth(text);
	uint16_t hlen = textWidth(headline);
	uint16_t wlen = tlen;
	if (hlen > tlen)
		wlen = hlen;
	uint16_t tx = (numButtons * 100) - 20;
	if (tx > wlen)
		wlen = tx;
	wlen += 100;
	if (wlen > 600)
		wlen = 600;
	const uint16_t headlineX = (SCREEN_W - hlen) / 2;
	const uint16_t textX = (SCREEN_W - tlen) / 2;
	const uint16_t x = (SCREEN_W - wlen) / 2;
	// the dialog's y position differs in extended pattern editor mode
	const uint16_t y = ui.extendedPatternEditor ? SYSTEM_REQUEST_Y_EXT : SYSTEM_REQUEST_Y;
	// set up buttons
	for (uint16_t i = 0; i < numButtons; i++)
	{
		pushButton_t *p = &pushButtons[i];
		p->x = ((SCREEN_W - tx) / 2) + (i * 100);
		p->y = y + 42;
		p->w = PUSHBUTTON_W;
		p->h = 16;
		p->caption = buttonText[type][i];
		p->visible = true;
	}
	// set up "don't show again" checkbox (if callback present)
	bool hasCheckbox = (checkBoxCallback != NULL);
	if (hasCheckbox)
	{
		checkBox_t *c = &checkBoxes[0];
		c->x = x + 5;
		c->y = y + 50;
		c->clickAreaWidth = 116;
		c->clickAreaHeight = 12;
		c->checked = false;
		c->callbackFunc = checkBoxCallback;
		c->visible = true;
	}
	mouse.lastUsedObjectType = OBJECT_NONE;
	mouse.lastUsedObjectID = OBJECT_ID_NONE;
	mouse.leftButtonPressed = 0;
	mouse.rightButtonPressed = 0;
	// input/rendering loop
	int16_t returnVal = 0;
	while (ui.sysReqShown)
	{
		beginFPSCounter();
		readMouseXY();
		setSyncedReplayerVars();
		if (mouse.leftButtonPressed || mouse.rightButtonPressed)
		{
			if (mouse.lastUsedObjectType == OBJECT_PUSHBUTTON)
				handlePushButtonsWhileMouseDown();
			else if (mouse.lastUsedObjectType == OBJECT_CHECKBOX)
				handleCheckBoxesWhileMouseDown();
		}
		while (SDL_PollEvent(&inputEvent))
		{
			handleWaitVblQuirk(&inputEvent);
			if (inputEvent.type == SDL_KEYDOWN)
			{
				if (inputEvent.key.keysym.sym == SDLK_ESCAPE)
				{
					if (!inputEvent.key.repeat) // don't let previously held-down ESC immediately close the box
					{
						returnVal = 0;
						ui.sysReqShown = false;
					}
				}
				else if (inputEvent.key.keysym.sym == SDLK_RETURN)
				{
					returnVal = 1;
					ui.sysReqShown = false;
					keyb.ignoreCurrKeyUp = true; // don't handle key up event for any keys that were pressed
				}
				for (uint16_t i = 0; i < numButtons; i++)
				{
					if (shortCut[type][i] == inputEvent.key.keysym.sym)
					{
						returnVal = i + 1;
						ui.sysReqShown = false;
						keyb.ignoreCurrKeyUp = true; // don't handle key up event for any keys that were pressed
						break;
					}
				}
			}
			else if (inputEvent.type == SDL_MOUSEBUTTONUP)
			{
				if (mouseButtonUpLogic(inputEvent.button.button))
				{
					if (hasCheckbox)
						testCheckBoxMouseRelease();
					returnVal = testPushButtonMouseRelease(false) + 1;
					if (returnVal > 0)
						ui.sysReqShown = false;
					mouse.lastUsedObjectID = OBJECT_ID_NONE;
					mouse.lastUsedObjectType = OBJECT_NONE;
				}
			}
			else if (inputEvent.type == SDL_MOUSEBUTTONDOWN)
			{
				if (mouseButtonDownLogic(inputEvent.button.button))
				{
					if (testPushButtonMouseDown()) continue;
					if (testCheckBoxMouseDown()) continue;
				}
			}
#if defined __APPLE__ && defined __aarch64__
			else if (inputEvent.type == SDL_MOUSEMOTION)
			{
				armMacGhostMouseCursorFix();
			}
#endif
			if (!ui.sysReqShown)
				break;
		}
		if (!ui.sysReqShown)
			break;
		handleRedrawing();
		// draw OK box
		drawWindow(wlen);
		textOutShadow(headlineX, y +  4, PAL_FORGRND, PAL_BUTTON2, headline);
		textOutShadow(textX,     y + 24, PAL_FORGRND, PAL_BUTTON2, text);
		for (uint16_t i = 0; i < numButtons; i++) drawPushButton(i);
		if (hasCheckbox)
		{
			drawCheckBox(0);
			textOutShadow(x + 21, y + 52, PAL_FORGRND, PAL_BUTTON2, "Don't show again");
		}
		flipFrame();
		endFPSCounter();
	}
	for (uint16_t i = 0; i < numButtons; i++)
		hidePushButton(i);
	if (hasCheckbox)
		hideCheckBox(0);
	mouse.lastUsedObjectID = oldLastUsedObjectID;
	mouse.lastUsedObjectType = oldLastUsedObjectType;
	unstuckLastUsedGUIElement();
	showBottomScreen();
	SDL_EventState(SDL_DROPFILE, SDL_ENABLE);
	return returnVal;
}
/* WARNING:
** - This routine must ONLY be called from the main input/video thread!!
** - edText must be NUL-terminated
*/
int16_t inputBox(int16_t type, const char *headline, char *edText, uint16_t maxStrLen)
{
#define PUSHBUTTON_W 80
#define TEXTBOX_W 250
	SDL_Event inputEvent;
	if (editor.editTextFlag)
		exitTextEditing();
	// revert "delete/rename" mouse modes (disk op.)
	if (mouse.mode != MOUSE_MODE_NORMAL)
		setMouseMode(MOUSE_MODE_NORMAL);
	if (ui.sysReqShown)
		return 0;
	int16_t oldLastUsedObjectID = mouse.lastUsedObjectID;
	int16_t oldLastUsedObjectType = mouse.lastUsedObjectType;
	textBox_t *t = &textBoxes[0];
	// set up text box
	memset(t, 0, sizeof (textBox_t));
	t->w = TEXTBOX_W;
	t->h = 12;
	t->tx = 2;
	t->ty = 1;
	t->textPtr = edText;
	t->maxChars = maxStrLen;
	t->changeMouseCursor = true;
	t->renderBufW = (9 + 1) * t->maxChars; // 9 = max character/glyph width possible
	t->renderBufH = 10; // 10 = max character height possible
	t->renderW = t->w - (t->tx * 2);
	t->renderBuf = (uint8_t *)malloc(t->renderBufW * t->renderBufH * sizeof (int8_t));
	if (t->renderBuf == NULL)
	{
		okBox(0, "System message", "Not enough memory!", NULL);
		return 0;
	}
	SDL_EventState(SDL_DROPFILE, SDL_DISABLE);
	ui.sysReqShown = true;
	mouseAnimOff();
	uint16_t wlen = textWidth(headline);
	const uint16_t headlineX = (SCREEN_W - wlen) / 2;
	// count number of buttons
	uint16_t numButtons = 0;
	while (buttonText[type][numButtons][0] != '\0' && numButtons < 5)
		numButtons++;
	uint16_t tx = TEXTBOX_W;
	if (tx > wlen)
		wlen = tx;
	tx = (numButtons * 100) - 20;
	if (tx > wlen)
		wlen = tx;
	wlen += 100;
	if (wlen > 600)
		wlen = 600;
	// the box y position differs in extended pattern editor mode
	const uint16_t y = ui.extendedPatternEditor ? SYSTEM_REQUEST_Y_EXT : SYSTEM_REQUEST_Y;
	// set further text box settings
	t->x = (SCREEN_W - TEXTBOX_W) / 2;
	t->y = y + 24;
	t->visible = true;
	// setup buttons
	pushButton_t *p = pushButtons;
	for (uint16_t i = 0; i < numButtons; i++, p++)
	{
		p->w = PUSHBUTTON_W;
		p->h = 16;
		p->x = ((SCREEN_W - tx) / 2) + (i * 100);
		p->y = y + 42;
		p->caption = buttonText[type][i];
		p->visible = true;
	}
	keyb.leftShiftPressed = false; // kludge
	setTextCursorToEnd(t);
	mouse.lastEditBox = 0;
	editor.editTextFlag = true;
	SDL_StartTextInput();
	mouse.lastUsedObjectType = OBJECT_NONE;
	mouse.lastUsedObjectID = OBJECT_ID_NONE;
	mouse.leftButtonPressed = 0;
	mouse.rightButtonPressed = 0;
	// input/rendering loop
	int16_t returnVal = 0;
	while (ui.sysReqShown)
	{
		beginFPSCounter();
		readMouseXY();
		readKeyModifiers();
		setSyncedReplayerVars();
		if (mouse.leftButtonPressed || mouse.rightButtonPressed)
		{
			if (mouse.lastUsedObjectType == OBJECT_PUSHBUTTON)
				handlePushButtonsWhileMouseDown();
			else if (mouse.lastUsedObjectType == OBJECT_TEXTBOX)
				handleTextBoxWhileMouseDown();
		}
		while (SDL_PollEvent(&inputEvent))
		{
			handleWaitVblQuirk(&inputEvent);
			if (inputEvent.type == SDL_TEXTINPUT)
			{
				if (editor.editTextFlag)
				{
					if (keyb.ignoreTextEditKey)
					{
						keyb.ignoreTextEditKey = false;
						continue;
					}
					char *inputText = utf8ToCp437(inputEvent.text.text, false);
					if (inputText != NULL)
					{
						if (inputText[0] != '\0')
							handleTextEditInputChar(inputText[0]);
						free(inputText);
					}
				}
			}
			else if (inputEvent.type == SDL_KEYDOWN)
			{
				if (inputEvent.key.keysym.sym == SDLK_ESCAPE)
				{
					returnVal = 0;
					ui.sysReqShown = false;
				}
				else if (inputEvent.key.keysym.sym == SDLK_RETURN)
				{
					returnVal = 1;
					ui.sysReqShown = false;
					keyb.ignoreCurrKeyUp = true; // don't handle key up event for any keys that were pressed
				}
				if (editor.editTextFlag)
				{
					handleTextEditControl(inputEvent.key.keysym.sym);
				}
				else
				{
					for (uint16_t i = 0; i < numButtons; i++)
					{
						if (shortCut[1][i] == inputEvent.key.keysym.sym)
						{
							returnVal = i + 1;
							ui.sysReqShown = false;
							keyb.ignoreCurrKeyUp = true; // don't handle key up event for any keys that were pressed
							break;
						}
					}
				}
			}
			else if (inputEvent.type == SDL_MOUSEBUTTONUP)
			{
				if (mouseButtonUpLogic(inputEvent.button.button))
				{
					returnVal = testPushButtonMouseRelease(false) + 1;
					if (returnVal > 0)
						ui.sysReqShown = false;
					mouse.lastUsedObjectID = OBJECT_ID_NONE;
					mouse.lastUsedObjectType = OBJECT_NONE;
				}
			}
			else if (inputEvent.type == SDL_MOUSEBUTTONDOWN)
			{
				if (mouseButtonDownLogic(inputEvent.button.button))
				{
					if (testTextBoxMouseDown()) continue;
					if (testPushButtonMouseDown()) continue;
				}
			}
#if defined __APPLE__ && defined __aarch64__
			else if (inputEvent.type == SDL_MOUSEMOTION)
			{
				armMacGhostMouseCursorFix();
			}
#endif
			if (!ui.sysReqShown)
				break;
		}
		if (!ui.sysReqShown)
			break;
		handleRedrawing();
		// draw input box
		drawWindow(wlen);
		textOutShadow(headlineX, y + 4, PAL_FORGRND, PAL_BUTTON2, headline);
		clearRect(t->x, t->y, t->w, t->h);
		hLine(t->x - 1,    t->y - 1,    t->w + 2, PAL_BUTTON2);
		vLine(t->x - 1,    t->y,        t->h + 1, PAL_BUTTON2);
		hLine(t->x,        t->y + t->h, t->w + 1, PAL_BUTTON1);
		vLine(t->x + t->w, t->y,        t->h,     PAL_BUTTON1);
		drawTextBox(0);
		for (uint16_t i = 0; i < numButtons; i++) drawPushButton(i);
		flipFrame();
		endFPSCounter();
	}
	editor.editTextFlag = false;
	SDL_StopTextInput();
	for (uint16_t i = 0; i < numButtons; i++)
		hidePushButton(i);
	hideTextBox(0);
	free(t->renderBuf);
	mouse.lastUsedObjectID = oldLastUsedObjectID;
	mouse.lastUsedObjectType = oldLastUsedObjectType;
	unstuckLastUsedGUIElement();
	showBottomScreen();
	SDL_EventState(SDL_DROPFILE, SDL_ENABLE);
	return returnVal;
}
// WARNING: This routine must NOT be called from the main input/video thread!
// If the checkBoxCallback argument is set, then you get a "Do not show again" checkbox.
int16_t okBoxThreadSafe(int16_t type, const char *headline, const char *text, void (*checkBoxCallback)(void))
{
	if (!editor.mainLoopOngoing)
		return 0; // main loop was not even started yet, bail out.
	// the amount of time to wait is not important, but close to one video frame makes sense
	const uint32_t waitTime = (uint32_t)((1000.0 / VBLANK_HZ) + 0.5);
	// block multiple calls before they are completed (just in case, for safety)
	while (okBoxData.active)
		SDL_Delay(waitTime);
	okBoxData.checkBoxCallback = checkBoxCallback;
	okBoxData.type = type;
	okBoxData.headline = headline;
	okBoxData.text = text;
	okBoxData.active = true;
	while (okBoxData.active)
		SDL_Delay(waitTime);
	return okBoxData.returnData;
}
static bool askQuit_RandomMsg(void)
{
	quitType_t *q = &quitMessage[rand() % QUIT_MESSAGES];
	int16_t button = okBox(q->type, "System request", q->text, NULL);
	return (button == 1) ? true : false;
}
bool askUnsavedChanges(uint8_t type)
{
	int16_t button;
	
	if (type == ASK_TYPE_QUIT)
	{
		button = okBox(2, "System request",
			"You have unsaved changes in your song. Do you still want to quit and lose ALL changes?", NULL);
	}
	else
	{
		button = okBox(2, "System request",
			"You have unsaved changes in your song. Load new song and lose ALL changes?", NULL);
	}
	return (button == 1) ? true : false;
}
int16_t quitBox(bool skipQuitMsg)
{
	if (ui.sysReqShown)
		return 0;
	if (!song.isModified && skipQuitMsg)
		return 1;
	if (song.isModified)
		return askUnsavedChanges(ASK_TYPE_QUIT);
	return askQuit_RandomMsg();
}