Kart-Public/tools/SRB2MP/SRB2MP.c
2014-03-15 13:11:35 -04:00

464 lines
9.7 KiB
C

// SRB2MP.cpp : Defines the entry point for the application.
//
#include "StdAfx.h"
#include "lump.h"
#define APPTITLE "SRB2 Music Player"
#define APPVERSION "v0.1"
#define APPAUTHOR "SSNTails"
#define APPCOMPANY "Sonic Team Junior"
static FSOUND_STREAM *fmus = NULL;
static int fsoundchannel = -1;
static FMUSIC_MODULE *mod = NULL;
static struct wadfile* wfptr = NULL;
static inline VOID M_SetVolume(int volume)
{
if (mod && FMUSIC_GetType(mod) != FMUSIC_TYPE_NONE)
FMUSIC_SetMasterVolume(mod, volume);
if (fsoundchannel != -1)
FSOUND_SetVolume(fsoundchannel, volume);
}
static inline BOOL M_InitMusic(VOID)
{
if (FSOUND_GetVersion() < FMOD_VERSION)
{
printf("Error : You are using the wrong DLL version!\nYou should be using FMOD %.02f\n", FMOD_VERSION);
return FALSE;
}
#ifdef _DEBUG
FSOUND_SetOutput(FSOUND_OUTPUT_WINMM);
#endif
if (!FSOUND_Init(44100, 1, FSOUND_INIT_GLOBALFOCUS))
{
printf("%s\n", FMOD_ErrorString(FSOUND_GetError()));
return FALSE;
}
return TRUE;
}
static inline VOID M_FreeMusic(VOID)
{
if (mod)
{
FMUSIC_StopSong(mod);
FMUSIC_FreeSong(mod);
mod = NULL;
}
if (fmus)
{
FSOUND_Stream_Stop(fmus);
FSOUND_Stream_Close(fmus);
fmus = NULL;
fsoundchannel = -1;
}
}
static inline VOID M_ShutdownMusic(VOID)
{
M_FreeMusic();
FSOUND_Close();
//remove(musicfile); // Delete the temp file
}
static inline VOID M_PauseMusic(VOID)
{
if (mod && !FMUSIC_GetPaused(mod))
FMUSIC_SetPaused(mod, TRUE);
if (fsoundchannel != -1 && FSOUND_IsPlaying(fsoundchannel))
FSOUND_SetPaused(fsoundchannel, TRUE);
}
static inline VOID M_ResumeMusic(VOID)
{
if (mod && FMUSIC_GetPaused(mod))
FMUSIC_SetPaused(mod, FALSE);
if (fsoundchannel != -1 && FSOUND_GetPaused(fsoundchannel))
FSOUND_SetPaused(fsoundchannel, FALSE);
}
static inline VOID M_StopMusic(VOID)
{
if (mod)
FMUSIC_StopSong(mod);
if (fsoundchannel != -1 && fmus && FSOUND_IsPlaying(fsoundchannel))
FSOUND_Stream_Stop(fmus);
}
static inline VOID M_StartFMODSong(LPVOID data, int len, int looping, HWND hDlg)
{
const int loops = FSOUND_LOOP_NORMAL|FSOUND_LOADMEMORY;
const int nloop = FSOUND_LOADMEMORY;
M_FreeMusic();
if (looping)
mod = FMUSIC_LoadSongEx(data, 0, len, loops, NULL, 0);
else
mod = FMUSIC_LoadSongEx(data, 0, len, nloop, NULL, 0);
if (mod)
{
FMUSIC_SetLooping(mod, (signed char)looping);
FMUSIC_SetPanSeperation(mod, 0.0f);
}
else
{
if (looping)
fmus = FSOUND_Stream_Open(data, loops, 0, len);
else
fmus = FSOUND_Stream_Open(data, nloop, 0, len);
}
if (!fmus && !mod)
{
MessageBoxA(hDlg, FMOD_ErrorString(FSOUND_GetError()), "Error", MB_OK|MB_APPLMODAL);
return;
}
// Scan the OGG for the COMMENT= field for a custom loop point
if (looping && fmus)
{
const BYTE *origpos, *dataum = data;
size_t scan, size = len;
CHAR buffer[512];
BOOL titlefound = FALSE, artistfound = FALSE;
unsigned int loopstart = 0;
origpos = dataum;
for(scan = 0; scan < size; scan++)
{
if (!titlefound)
{
if (*dataum++ == 'T')
if (*dataum++ == 'I')
if (*dataum++ == 'T')
if (*dataum++ == 'L')
if (*dataum++ == 'E')
if (*dataum++ == '=')
{
size_t titlecount = 0;
CHAR title[256];
BYTE length = *(dataum-10) - 6;
while(titlecount < length)
title[titlecount++] = *dataum++;
title[titlecount] = '\0';
sprintf(buffer, "Title: %s", title);
SendMessage(GetDlgItem(hDlg, IDC_TITLE), WM_SETTEXT, 0, (LPARAM)(LPCSTR)buffer);
titlefound = TRUE;
}
}
}
dataum = origpos;
for(scan = 0; scan < size; scan++)
{
if (!artistfound)
{
if (*dataum++ == 'A')
if (*dataum++ == 'R')
if (*dataum++ == 'T')
if (*dataum++ == 'I')
if (*dataum++ == 'S')
if (*dataum++ == 'T')
if (*dataum++ == '=')
{
size_t artistcount = 0;
CHAR artist[256];
BYTE length = *(dataum-11) - 7;
while(artistcount < length)
artist[artistcount++] = *dataum++;
artist[artistcount] = '\0';
sprintf(buffer, "By: %s", artist);
SendMessage(GetDlgItem(hDlg, IDC_ARTIST), WM_SETTEXT, 0, (LPARAM)(LPCSTR)buffer);
artistfound = TRUE;
}
}
}
dataum = origpos;
for(scan = 0; scan < size; scan++)
{
if (*dataum++ == 'C'){
if (*dataum++ == 'O'){
if (*dataum++ == 'M'){
if (*dataum++ == 'M'){
if (*dataum++ == 'E'){
if (*dataum++ == 'N'){
if (*dataum++ == 'T'){
if (*dataum++ == '='){
if (*dataum++ == 'L'){
if (*dataum++ == 'O'){
if (*dataum++ == 'O'){
if (*dataum++ == 'P'){
if (*dataum++ == 'P'){
if (*dataum++ == 'O'){
if (*dataum++ == 'I'){
if (*dataum++ == 'N'){
if (*dataum++ == 'T'){
if (*dataum++ == '=')
{
size_t newcount = 0;
CHAR looplength[64];
while (*dataum != 1 && newcount < 63)
{
looplength[newcount++] = *dataum++;
}
looplength[newcount] = '\n';
loopstart = atoi(looplength);
}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
else
dataum--;}
}
if (loopstart > 0)
{
const int length = FSOUND_Stream_GetLengthMs(fmus);
const int freq = 44100;
const unsigned int loopend = (unsigned int)((freq/1000.0f)*length-(freq/1000.0f));
if (!FSOUND_Stream_SetLoopPoints(fmus, loopstart, loopend))
{
printf("FMOD(Start,FSOUND_Stream_SetLoopPoints): %s\n",
FMOD_ErrorString(FSOUND_GetError()));
}
}
}
if (mod)
FMUSIC_PlaySong(mod);
if (fmus)
fsoundchannel = FSOUND_Stream_PlayEx(FSOUND_FREE, fmus, NULL, FALSE);
M_SetVolume(128);
}
static inline VOID FreeWADLumps(VOID)
{
M_FreeMusic();
if (wfptr) free_wadfile(wfptr);
wfptr = NULL;
}
static inline VOID ReadWADLumps(LPCSTR WADfilename, HWND hDlg)
{
HWND listbox = GetDlgItem(hDlg, IDC_PLAYLIST);
FILE* f;
struct lumplist *curlump;
SendMessage(listbox, LB_RESETCONTENT, 0, 0);
FreeWADLumps();
f = fopen(WADfilename, "rb");
wfptr = read_wadfile(f);
fclose(f);
/* start of C_LIST */
/* Loop through the lump list, printing lump info */
for(curlump = wfptr->head->next; curlump; curlump = curlump->next)
{
LPCSTR lumpname = get_lump_name(curlump->cl);
if (!strncmp(lumpname, "O_", 2) || !strncmp(lumpname, "D_", 2))
SendMessage(listbox, LB_ADDSTRING, 0, (LPARAM)(LPCSTR)get_lump_name(curlump->cl));
}
/* end of C_LIST */
}
//
// OpenWadfile
//
// Provides a common dialog box
// for selecting the desired wad file.
//
static inline VOID OpenWadfile(HWND hDlg)
{
OPENFILENAMEA ofn;
CHAR FileBuffer[256] = "";
ZeroMemory(&ofn, sizeof(ofn));
ofn.hwndOwner = hDlg;
ofn.lpstrFilter = "WAD Files\0*.wad\0All Files\0*.*\0\0";
ofn.lpstrInitialDir = NULL;
ofn.lpstrFile = FileBuffer;
ofn.lStructSize = sizeof(ofn);
ofn.nMaxFile = sizeof(FileBuffer);
ofn.nFilterIndex = 1;
ofn.lpstrFileTitle = NULL;
ofn.nMaxFileTitle = 0;
ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST | OFN_HIDEREADONLY;
if (GetOpenFileNameA(&ofn))
ReadWADLumps(FileBuffer, hDlg);
}
static inline VOID PlayLump(HWND hDlg)
{
HWND listbox = GetDlgItem(hDlg, IDC_PLAYLIST);
LRESULT cursel = SendMessage(listbox, LB_GETCURSEL, 0, 0);
/* Start of C_EXTRACT */
CHAR argv[9];
SendMessage(listbox, LB_GETTEXT, cursel, (LPARAM)(LPCSTR)argv);
/* Extract LUMPNAME FILENAME pairs */
if (wfptr)
{
struct lumplist *extracted;
printf("Extracting lump %s...\n", argv);
/* Find the lump to extract */
extracted = find_previous_lump(wfptr->head, NULL, argv);
if (extracted == NULL || (extracted = extracted->next) == NULL)
return;
/* Extract lump */
M_StartFMODSong(extracted->cl->data, extracted->cl->len, FALSE, hDlg);
/* end of extracting LUMPNAME FILENAME pairs */
} /* end of C_EXTRACT */
return;
}
static LRESULT CALLBACK MainWndproc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
return DefWindowProc(hWnd, message, wParam, lParam);
}
static INT_PTR CALLBACK DialogProc(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
switch(message)
{
case WM_INITDIALOG:
M_InitMusic();
break;
case WM_CLOSE:
EndDialog(hDlg, message);
break;
case WM_DESTROY:
FreeWADLumps();
M_ShutdownMusic();
break;
case WM_COMMAND:
{
switch(LOWORD(wParam))
{
case 2:
EndDialog(hDlg, message);
return TRUE;
case IDC_ABOUT: // The About button.
{
char TempString[256];
sprintf(TempString, "%s %s by %s - %s", APPTITLE, APPVERSION, APPAUTHOR, APPCOMPANY);
MessageBoxA(hDlg, TempString, "About", MB_OK|MB_APPLMODAL);
}
return TRUE;
case IDC_OPEN:
OpenWadfile(hDlg);
return TRUE;
case IDC_PLAY:
PlayLump(hDlg);
return TRUE;
default:
break;
}
break;
}
}
return FALSE;
}
static inline VOID RegisterDialogClass(LPCSTR name, WNDPROC callback, HINSTANCE hInst)
{
WNDCLASSA wnd;
wnd.style = CS_HREDRAW | CS_VREDRAW;
wnd.cbWndExtra = DLGWINDOWEXTRA;
wnd.cbClsExtra = 0;
wnd.hCursor = LoadCursorA(NULL,(LPCSTR)MAKEINTRESOURCE(IDC_ARROW));
wnd.hIcon = LoadIcon(NULL,MAKEINTRESOURCE(IDI_ICON1));
wnd.hInstance = hInst;
wnd.lpfnWndProc = callback;
wnd.lpszClassName = name;
wnd.lpszMenuName = NULL;
wnd.hbrBackground = (HBRUSH)(COLOR_WINDOW);
if (!RegisterClassA(&wnd))
{
return;
}
}
int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
// Prevent multiples instances of this app.
CreateMutexA(NULL, TRUE, APPTITLE);
if (GetLastError() == ERROR_ALREADY_EXISTS)
return 0;
RegisterDialogClass("SRB2MP", MainWndproc, hInstance);
DialogBoxA(hInstance, (LPCSTR)IDD_MAIN, NULL, DialogProc);
return 0;
}