Descent3/editor/ScriptSyncDialog.cpp
2024-06-15 20:12:48 +02:00

482 lines
12 KiB
C++

/*
* Descent 3
* Copyright (C) 2024 Parallax Software
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*/
// ScriptSyncDialog.cpp : implementation file
//
#include "mfc_compatibility.h"
#include "editor.h"
#include "ScriptSyncDialog.h"
#include "ddio.h"
#include "cfile.h"
#include "psglob.h"
#include "gamefile.h"
#include "manage.h"
#include "ScriptCompilerAPI.h"
#include "mem.h"
#include "mono.h"
#include "textaux.h"
#include "appdatabase.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
bool ManageFindFirst(char *buffer, char *wildcard);
bool ManageFindNext(char *buffer);
void ManageFindClose(void);
uint8_t DetermineScriptType(char *filename);
// returns true if the given .cpp file is has an out of date dll (or non-existant). Working
// dir MUST be set to data\scripts before calling this function
bool IsScriptOutofDate(char *name);
// returns true if the given .cpp/.dll file is out of sync (or non-existant). Working directory
// doesn't necessarily have to be set to data\scripts.
bool IsScriptOutofSync(char *name);
/////////////////////////////////////////////////////////////////////////////
// CScriptSyncDialog dialog
CScriptSyncDialog::CScriptSyncDialog(CWnd *pParent /*=NULL*/) : CDialog(CScriptSyncDialog::IDD, pParent) {
//{{AFX_DATA_INIT(CScriptSyncDialog)
// NOTE: the ClassWizard will add member initialization here
//}}AFX_DATA_INIT
}
void CScriptSyncDialog::DoDataExchange(CDataExchange *pDX) {
CDialog::DoDataExchange(pDX);
//{{AFX_DATA_MAP(CScriptSyncDialog)
DDX_Control(pDX, IDC_LIST, m_List);
//}}AFX_DATA_MAP
}
BEGIN_MESSAGE_MAP(CScriptSyncDialog, CDialog)
//{{AFX_MSG_MAP(CScriptSyncDialog)
ON_WM_PAINT()
ON_WM_TIMER()
ON_WM_DESTROY()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CScriptSyncDialog message handlers
/////////////////////////////////////////////////////////////////////////////
// CMyListCtrl
CMyListCtrl::CMyListCtrl() {}
CMyListCtrl::~CMyListCtrl() {}
BEGIN_MESSAGE_MAP(CMyListCtrl, CListCtrl)
//{{AFX_MSG_MAP(CMyListCtrl)
// NOTE - the ClassWizard will add and remove mapping macros here.
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CMyListCtrl message handlers
#define COLUMN_NAME 0
#define COLUMN_STATE 1
#define COLUMN_NAME_SIZE 0.49f
#define COLUMN_STATE_SIZE 0.49f
void CMyListCtrl::Initialize(void) {
ModifyStyleEx(0, LVS_EX_FULLROWSELECT);
RECT rect;
GetWindowRect(&rect);
int width = rect.right - rect.left;
int ret;
ret = InsertColumn(COLUMN_NAME, "Name", LVCFMT_LEFT, (int)(width * COLUMN_NAME_SIZE));
ret = InsertColumn(COLUMN_STATE, "State", LVCFMT_LEFT, (int)(width * COLUMN_STATE_SIZE));
}
int CMyListCtrl::AddItem(char *name, int num) {
int item_count = GetItemCount();
int added = InsertItem(item_count, name);
if (added == -1) {
return -1;
}
bool ret = CMyListCtrl::SetItem(added, num);
if (!ret) {
DeleteItem(added);
return -1;
}
// all went ok
return added;
}
bool CMyListCtrl::GetItem(int item, char *name, char *state) {
int item_count = GetItemCount();
if (item >= item_count)
return false;
GetItemText(item, COLUMN_NAME, name, 256);
GetItemText(item, COLUMN_STATE, state, 40);
return true;
}
bool CMyListCtrl::SetItem(int item, int state) {
char buffer[256];
switch (state) {
case STATE_COMPILED:
strcpy(buffer, "Compiled");
break;
case STATE_OOD:
strcpy(buffer, "Out of Date");
break;
case STATE_ERROOD:
strcpy(buffer, "Error/Out of Date");
break;
case STATE_OOS:
strcpy(buffer, "Out of Sync");
break;
case STATE_ERROOS:
strcpy(buffer, "Error/Out of Sync");
break;
case STATE_MISSING:
strcpy(buffer, "Missing");
break;
case STATE_NOCOMPILER:
strcpy(buffer, "No Compiler");
break;
}
return (CListCtrl::SetItem(item, COLUMN_STATE, LVIF_TEXT, buffer, 0, 0, 0, 0) != FALSE);
}
void CMyListCtrl::SelectItem(int index) {
int item_count = GetItemCount();
if (index < 0 || index >= item_count)
return;
LVITEM lv;
lv.mask = LVIF_STATE;
lv.iSubItem = 0;
lv.stateMask = LVIS_SELECTED;
lv.state = 0;
for (int i = 0; i < item_count; i++) {
lv.iItem = i;
CListCtrl::SetItem(&lv);
}
lv.iItem = index;
lv.state = 0xFFFFFFFF;
CListCtrl::SetItem(&lv);
}
class tManageFindIn {
public:
tManageFindIn() { searching = false; }
bool searching;
bool in_Gamefiles;
int curr_index;
char glob_string[PAGENAME_LEN];
};
tManageFindIn ManageFindInData;
bool ManageFindFirst(char *buffer, char *wildcard) {
if (ManageFindInData.searching)
ManageFindClose();
ManageFindInData.searching = true;
strcpy(ManageFindInData.glob_string, wildcard);
ManageFindInData.curr_index = 0;
ManageFindInData.in_Gamefiles = true;
return ManageFindNext(buffer);
}
bool ManageFindNext(char *buffer) {
if (!ManageFindInData.searching)
return false;
if (ManageFindInData.in_Gamefiles) {
for (; ManageFindInData.curr_index < MAX_GAMEFILES; ManageFindInData.curr_index++) {
if (Gamefiles[ManageFindInData.curr_index].used &&
PSGlobMatch(ManageFindInData.glob_string, Gamefiles[ManageFindInData.curr_index].name, false, false)) {
// match
strcpy(buffer, Gamefiles[ManageFindInData.curr_index].name);
ManageFindInData.curr_index++;
return true;
}
}
ManageFindInData.curr_index = 0;
ManageFindInData.in_Gamefiles = false;
}
for (; ManageFindInData.curr_index < MAX_TRACKLOCKS; ManageFindInData.curr_index++) {
if (GlobalTrackLocks[ManageFindInData.curr_index].used &&
GlobalTrackLocks[ManageFindInData.curr_index].pagetype == PAGETYPE_GAMEFILE &&
PSGlobMatch(ManageFindInData.glob_string, GlobalTrackLocks[ManageFindInData.curr_index].name, false, false)) {
// match
strcpy(buffer, GlobalTrackLocks[ManageFindInData.curr_index].name);
ManageFindInData.curr_index++;
return true;
}
}
return false;
}
void ManageFindClose(void) { ManageFindInData.searching = false; }
/////////////////////////////////////////////////////////////////
BOOL CScriptSyncDialog::OnInitDialog() {
CDialog::OnInitDialog();
early_exit = false;
first_paint = true;
m_InStuff = false;
m_Done = false;
m_Index = 0;
m_NumFiles = 0;
m_HasCompiler = -1;
m_Files = NULL;
m_FillingIn = true;
m_ErrorCount = 0;
ddio_GetWorkingDir(old_dir, _MAX_PATH);
ddio_SetWorkingDir(LocalScriptDir);
m_List.Initialize();
BuildList();
return TRUE;
}
void CScriptSyncDialog::defer(void) {
MSG msg;
while (1) {
if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
if (msg.message == WM_QUIT) {
// QUIT APP.
early_exit = true;
}
TranslateMessage(&msg);
DispatchMessage(&msg);
} else {
break;
}
}
}
void CScriptSyncDialog::BuildList(void) {
char buffer[256];
m_NumFiles = 0;
m_Files = NULL;
if (ManageFindFirst(buffer, "*.cpp")) {
m_NumFiles++;
while (ManageFindNext(buffer)) {
m_NumFiles++;
}
}
ManageFindClose();
if (m_NumFiles == 0)
return;
m_Files = (tFileInfo *)mem_malloc(sizeof(tFileInfo) * m_NumFiles);
m_NumFiles = 0;
if (ManageFindFirst(buffer, "*.cpp")) {
strcpy(m_Files[m_NumFiles].filename, buffer);
m_NumFiles++;
while (ManageFindNext(buffer)) {
strcpy(m_Files[m_NumFiles].filename, buffer);
m_NumFiles++;
}
}
ManageFindClose();
}
CString SS_CompileOutputData;
void SS_CompileOutputCallback(char *str) { SS_CompileOutputData += str; }
void CScriptSyncDialog::DoStuff(void) {
m_InStuff = true;
defer();
if (m_Index >= m_NumFiles) {
if (m_FillingIn) {
m_FillingIn = false;
m_Index = 0;
} else {
m_Done = true;
goto done;
}
}
if (m_FillingIn) {
char dllfilename[_MAX_PATH];
ddio_SplitPath(m_Files[m_Index].filename, NULL, dllfilename, NULL);
strcat(dllfilename, ".dll");
if (!cfexist(dllfilename)) {
m_Files[m_Index].state = STATE_MISSING;
} else {
if (IsScriptOutofSync(m_Files[m_Index].filename)) {
m_Files[m_Index].state = STATE_OOS;
} else if (IsScriptOutofDate(m_Files[m_Index].filename)) {
m_Files[m_Index].state = STATE_OOD;
} else
m_Files[m_Index].state = STATE_COMPILED;
}
m_List.AddItem(m_Files[m_Index].filename, m_Files[m_Index].state);
m_List.EnsureVisible(m_Index, false);
} else {
if (m_Files) {
switch (m_Files[m_Index].state) {
case STATE_COMPILED:
break;
case STATE_MISSING:
case STATE_OOD:
case STATE_OOS: {
char Compiler_path[_MAX_PATH];
int len = _MAX_PATH;
// attempt to recompile the script
// make sure there is a compiler defined
if (m_HasCompiler == -1) {
if (Database->read("EditorCompiler", Compiler_path, &len)) {
if (!cfexist(Compiler_path)) {
m_HasCompiler = 0;
} else {
m_HasCompiler = 1;
}
} else {
m_HasCompiler = 0;
}
}
if (m_HasCompiler) {
tCompilerInfo ci;
char dllfilename[_MAX_PATH];
SS_CompileOutputData = "";
m_List.EnsureVisible(m_Index, false);
m_List.SelectItem(m_Index);
defer();
ci.callback = SS_CompileOutputCallback;
ci.script_type = DetermineScriptType(m_Files[m_Index].filename);
strcpy(ci.source_filename, m_Files[m_Index].filename);
ScriptCompile(&ci);
ddio_SplitPath(m_Files[m_Index].filename, NULL, dllfilename, NULL);
strcat(dllfilename, ".dll");
if (!cfexist(dllfilename)) {
m_Files[m_Index].state = STATE_MISSING;
} else {
if (IsScriptOutofSync(m_Files[m_Index].filename)) {
m_Files[m_Index].state = STATE_ERROOS;
} else if (IsScriptOutofDate(m_Files[m_Index].filename)) {
m_Files[m_Index].state = STATE_ERROOD;
} else
m_Files[m_Index].state = STATE_COMPILED;
}
if (m_Files[m_Index].state == STATE_MISSING || m_Files[m_Index].state == STATE_ERROOD ||
m_Files[m_Index].state == STATE_ERROOS) {
m_ErrorCount++;
char *string;
string = mem_strdup(SS_CompileOutputData.GetBuffer(0));
if (string) {
char buffer[2048];
const char *next_line;
mprintf(0, "===========================================\n");
next_line = textaux_CopyTextLine(string, buffer);
while (next_line) {
mprintf(0, "%s\n", buffer);
next_line = textaux_CopyTextLine(next_line, buffer);
}
mprintf(0, "===========================================\n");
mem_free(string);
}
}
} else {
m_Files[m_Index].state = STATE_NOCOMPILER;
}
m_List.SetItem(m_Index, m_Files[m_Index].state);
} break;
}
}
}
done:
m_Index++;
m_InStuff = false;
}
void CScriptSyncDialog::OnPaint() {
CPaintDC dc(this); // device context for painting
if (first_paint) {
first_paint = false;
SetTimer(0x12, 200, NULL);
}
}
void CScriptSyncDialog::OnTimer(UINT nIDEvent) {
defer();
if (m_Done) {
CDialog::OnOK();
} else {
if (!m_InStuff && nIDEvent == 0x12) {
DoStuff();
}
}
CDialog::OnTimer(nIDEvent);
if (!m_Done)
SetTimer(0x12, 50, NULL);
}
void CScriptSyncDialog::OnDestroy() {
CDialog::OnDestroy();
if (m_Files) {
mem_free(m_Files);
}
ddio_SetWorkingDir(old_dir);
}