// Copyright (C) 2003-2008 Dolphin Project. // 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, version 2.0. // 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 2.0 for more details. // A copy of the GPL 2.0 should have been included with the program. // If not, see http://www.gnu.org/licenses/ // Official SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ #include "MemcardManager.h" #include "Common.h" #include "wx/mstream.h" #define DEFAULTS wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator #define ARROWS slot ? _T("") : ARROW[slot], slot ? ARROW[slot] : _T("") const u8 hdr[] = { 0x42,0x4D, 0x38,0x30,0x00,0x00, 0x00,0x00,0x00,0x00, 0x36,0x00,0x00,0x00, 0x28,0x00,0x00,0x00, 0x20,0x00,0x00,0x00, //W 0x20,0x00,0x00,0x00, //H 0x01,0x00, 0x20,0x00, 0x00,0x00,0x00,0x00, 0x02,0x30,0x00,0x00, //data size 0x12,0x0B,0x00,0x00, 0x12,0x0B,0x00,0x00, 0x00,0x00,0x00,0x00, 0x00,0x00,0x00,0x00 }; wxBitmap wxBitmapFromMemoryRGBA(const unsigned char* data, int width, int height) { int stride = (4*width); int bytes = (stride*height) + sizeof(hdr); bytes = (bytes+3)&(~3); u8 *pdata = new u8[bytes]; memset(pdata,0,bytes); memcpy(pdata,hdr,sizeof(hdr)); u8 *pixelData = pdata + sizeof(hdr); for (int y=0;yDestroy(); delete MemcardManagerDebug; MemcardManagerDebug = NULL; } #endif MemcardManagerIni.Load(CONFIG_FILE); MemcardManagerIni.Set("MemcardManager", "Items per page", itemsPerPage); if (!DefaultMemcard[SLOT_A].empty() && (strcmp(DefaultMemcard[SLOT_A].c_str(), "."))) MemcardManagerIni.Set("MemcardManager", "DefaultMemcardA", DefaultMemcard[SLOT_A]); else MemcardManagerIni.DeleteKey("MemcardManager", "DefaultMemcardA"); if (!DefaultMemcard[SLOT_B].empty() && (strcmp(DefaultMemcard[SLOT_B].c_str(), "."))) MemcardManagerIni.Set("MemcardManager", "DefaultMemcardB", DefaultMemcard[SLOT_B]); else MemcardManagerIni.DeleteKey("MemcardManager", "DefaultMemcardB"); MemcardManagerIni.Save(CONFIG_FILE); } CMemcardManager::CMemcardListCtrl::CMemcardListCtrl(wxWindow* parent, const wxWindowID id, const wxPoint& pos, const wxSize& size, long style) : wxListCtrl(parent, id, pos, size, style) { if (MemcardManagerIni.Load(CONFIG_FILE)) { MemcardManagerIni.Get("MemcardManager", "Use Pages", &usePages, true); MemcardManagerIni.Get("MemcardManager", "cBanner", &column[COLUMN_BANNER], true); MemcardManagerIni.Get("MemcardManager", "cTitle", &column[COLUMN_TITLE], true); MemcardManagerIni.Get("MemcardManager", "cComment", &column[COLUMN_COMMENT], true); MemcardManagerIni.Get("MemcardManager", "cIcon", &column[COLUMN_ICON], true); MemcardManagerIni.Get("MemcardManager", "cBlocks", &column[COLUMN_BLOCKS], true); MemcardManagerIni.Get("MemcardManager", "cFirst Block", &column[COLUMN_FIRSTBLOCK], true); #ifdef DEBUG_MCM MemcardManagerIni.Get("MemcardManager", "cDebug", &column[NUMBER_OF_COLUMN], false); #else column[NUMBER_OF_COLUMN] = false; #endif for(int i = COLUMN_GAMECODE; i < NUMBER_OF_COLUMN; i++) { column[i] = column[NUMBER_OF_COLUMN]; } } else { usePages = true; for (int i = 0; i < NUMBER_OF_COLUMN; i++) { if ( i > COLUMN_FIRSTBLOCK) column[i] = false; else column[i] = true; } } twoCardsLoaded = false; prevPage = false; nextPage = false; } CMemcardManager::CMemcardListCtrl::~CMemcardListCtrl() { MemcardManagerIni.Load(CONFIG_FILE); MemcardManagerIni.Set("MemcardManager", "Use Pages", usePages); MemcardManagerIni.Set("MemcardManager", "cBanner", column[COLUMN_BANNER]); MemcardManagerIni.Set("MemcardManager", "cTitle", column[COLUMN_TITLE]); MemcardManagerIni.Set("MemcardManager", "cComment", column[COLUMN_COMMENT]); MemcardManagerIni.Set("MemcardManager", "cIcon", column[COLUMN_ICON]); MemcardManagerIni.Set("MemcardManager", "cBlocks", column[COLUMN_BLOCKS]); MemcardManagerIni.Set("MemcardManager", "cFirst Block", column[COLUMN_FIRSTBLOCK]); #ifdef DEBUG_MCM MemcardManagerIni.Set("MemcardManager", "cDebug", column[NUMBER_OF_COLUMN]); #endif MemcardManagerIni.Save(CONFIG_FILE); } void CMemcardManager::CreateGUIControls() { // Create the controls for both memcards const wxChar* ARROW[2] = {_T("<-"), _T("->")}; m_ConvertToGci = new wxButton(this, ID_CONVERTTOGCI, wxT("Convert to GCI"), DEFAULTS); for (int slot = SLOT_A; slot < SLOT_B + 1; slot++) { m_CopyFrom[slot] = new wxButton(this, ID_COPYFROM_A + slot, wxString::Format(wxT("%1$sCopy%1$s"), ARROW[slot ? 0 : 1]), DEFAULTS); m_SaveImport[slot] = new wxButton(this, ID_SAVEIMPORT_A + slot, wxString::Format(wxT("%sImport GCI%s"), ARROWS), DEFAULTS); m_SaveExport[slot] = new wxButton(this, ID_SAVEEXPORT_A + slot, wxString::Format(wxT("%sExport GCI%s"), ARROWS), DEFAULTS); m_Delete[slot] = new wxButton(this, ID_DELETE_A + slot, wxString::Format(wxT("%sDelete%s"), ARROWS), DEFAULTS); m_PrevPage[slot] = new wxButton(this, ID_PREVPAGE_A + slot, wxT("Prev Page"), DEFAULTS); m_NextPage[slot] = new wxButton(this, ID_NEXTPAGE_A + slot, wxT("Next Page"), DEFAULTS); t_Status[slot] = new wxStaticText(this, 0, wxEmptyString, wxDefaultPosition,wxDefaultSize, 0, wxEmptyString); sPages[slot] = new wxBoxSizer(wxHORIZONTAL); sPages[slot]->Add(m_PrevPage[slot], 0, wxEXPAND|wxALL, 1); sPages[slot]->Add(t_Status[slot],0, wxEXPAND|wxALL, 5); sPages[slot]->Add(0, 0, 1, wxEXPAND|wxALL, 0); sPages[slot]->Add(m_NextPage[slot], 0, wxEXPAND|wxALL, 1); m_MemcardPath[slot] = new wxFilePickerCtrl(this, ID_MEMCARDPATH_A + slot, wxT(FULL_GC_USER_DIR), wxT("Choose a memory card:"), wxT("Gamecube Memory Cards (*.raw,*.gcp)|*.raw;*.gcp"), wxDefaultPosition, wxDefaultSize, wxFLP_USE_TEXTCTRL|wxFLP_OPEN); m_MemcardList[slot] = new CMemcardListCtrl(this, ID_MEMCARDLIST_A + slot, wxDefaultPosition, wxSize(350,400), wxLC_REPORT | wxSUNKEN_BORDER | wxLC_ALIGN_LEFT | wxLC_SINGLE_SEL); m_MemcardList[slot]->AssignImageList(new wxImageList(96,32),wxIMAGE_LIST_SMALL); sMemcard[slot] = new wxStaticBoxSizer(wxVERTICAL, this, wxString::Format(wxT("Memory Card %c"), 'A' + slot)); sMemcard[slot]->Add(m_MemcardPath[slot], 0, wxEXPAND|wxALL, 5); sMemcard[slot]->Add(m_MemcardList[slot], 1, wxEXPAND|wxALL, 5); sMemcard[slot]->Add(sPages[slot], 0, wxEXPAND|wxALL, 1); } sButtons = new wxBoxSizer(wxVERTICAL); sButtons->AddStretchSpacer(2); sButtons->Add(m_CopyFrom[SLOT_B], 0, wxEXPAND, 5); sButtons->Add(m_CopyFrom[SLOT_A], 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_SaveImport[SLOT_A], 0, wxEXPAND, 5); sButtons->Add(m_SaveExport[SLOT_A], 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_ConvertToGci, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_SaveImport[SLOT_B], 0, wxEXPAND, 5); sButtons->Add(m_SaveExport[SLOT_B], 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_Delete[SLOT_A], 0, wxEXPAND, 5); sButtons->Add(m_Delete[SLOT_B], 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sMain = new wxBoxSizer(wxHORIZONTAL); sMain->Add(sMemcard[SLOT_A], 1, wxEXPAND|wxALL, 5); sMain->Add(sButtons, 0, wxEXPAND, 0); sMain->Add(sMemcard[SLOT_B], 1, wxEXPAND|wxALL, 5); this->SetSizer(sMain); sMain->SetSizeHints(this); Fit(); for (int i = SLOT_A; i < SLOT_B + 1; i++) { m_PrevPage[i]->Disable(); m_NextPage[i]->Disable(); m_CopyFrom[i]->Disable(); m_SaveImport[i]->Disable(); m_SaveExport[i]->Disable(); m_Delete[i]->Disable(); if (strcmp(DefaultMemcard[i].c_str(), ".")) { m_MemcardPath[i]->SetPath(wxString::FromAscii(DefaultMemcard[i].c_str())); ChangePath(ID_MEMCARDPATH_A + i); } } } void CMemcardManager::OnClose(wxCloseEvent& WXUNUSED (event)) { Destroy(); } void CMemcardManager::OnPathChange(wxFileDirPickerEvent& event) { ChangePath(event.GetId()); } void CMemcardManager::ChangePath(int id) { int slot = SLOT_B; int slot2 = SLOT_A; switch (id) { case ID_MEMCARDPATH_A: slot = SLOT_A; slot2 = SLOT_B; case ID_MEMCARDPATH_B: page[slot] = FIRSTPAGE; if (m_MemcardList[slot]->usePages && m_PrevPage[slot]->IsEnabled()) { m_PrevPage[slot]->Disable(); m_MemcardList[slot]->prevPage = false; } if (!strcasecmp(m_MemcardPath[slot2]->GetPath().mb_str(), m_MemcardPath[slot]->GetPath().mb_str())) { if(!m_MemcardPath[slot]->GetPath().IsEmpty()) PanicAlert("Memcard already opened"); } else if (ReloadMemcard(m_MemcardPath[slot]->GetPath().mb_str(), slot)) { m_MemcardList[slot2]->twoCardsLoaded = true; m_SaveImport[slot]->Enable(); m_SaveExport[slot]->Enable(); m_Delete[slot]->Enable(); break; } if (memoryCard[slot]) { delete memoryCard[slot]; memoryCard[slot] = NULL; } m_MemcardList[slot2]->twoCardsLoaded = false; m_MemcardPath[slot]->SetPath(wxEmptyString); m_MemcardList[slot]->ClearAll(); t_Status[slot]->SetLabel(wxEmptyString); m_SaveImport[slot]->Disable(); m_SaveExport[slot]->Disable(); m_Delete[slot]->Disable(); if (m_MemcardList[slot]->usePages) { m_PrevPage[slot]->Disable(); m_NextPage[slot]->Disable(); } break; } if (m_Delete[SLOT_A]->IsEnabled() && m_Delete[SLOT_B]->IsEnabled()) { m_CopyFrom[SLOT_A]->Enable(); m_CopyFrom[SLOT_B]->Enable(); } else { m_CopyFrom[SLOT_A]->Disable(); m_CopyFrom[SLOT_B]->Disable(); } } void CMemcardManager::OnPageChange(wxCommandEvent& event) { int slot = SLOT_B; switch (event.GetId()) { case ID_NEXTPAGE_A: slot = SLOT_A; case ID_NEXTPAGE_B: if (!m_PrevPage[slot]->IsEnabled()) { m_PrevPage[slot]->Enable(); m_MemcardList[slot]->prevPage = true; } page[slot]++; if (page[slot] == maxPages) { m_NextPage[slot]->Disable(); m_MemcardList[slot]->nextPage = false; } ReloadMemcard(m_MemcardPath[slot]->GetPath().mb_str(), slot); break; case ID_PREVPAGE_A: slot = SLOT_A; case ID_PREVPAGE_B: if (!m_NextPage[slot]->IsEnabled()) { m_NextPage[slot]->Enable(); m_MemcardList[slot]->nextPage = true; } page[slot]--; if (!page[slot]) { m_PrevPage[slot]->Disable(); m_MemcardList[slot]->prevPage = false; } ReloadMemcard(m_MemcardPath[slot]->GetPath().mb_str(), slot); break; } } void CMemcardManager::OnMenuChange(wxCommandEvent& event) { switch(event.GetId()) { case ID_USEPAGES: m_MemcardList[SLOT_A]->usePages = !m_MemcardList[SLOT_A]->usePages; m_MemcardList[SLOT_B]->usePages = !m_MemcardList[SLOT_B]->usePages; if (!m_MemcardList[SLOT_A]->usePages) { m_PrevPage[SLOT_A]->Disable(); m_PrevPage[SLOT_B]->Disable(); m_NextPage[SLOT_A]->Disable(); m_NextPage[SLOT_B]->Disable(); page[SLOT_A] = page[SLOT_B] = FIRSTPAGE; } break; case ID_MEMCARDPATH_A: DefaultMemcard[SLOT_A] = m_MemcardPath[SLOT_A]->GetPath().mb_str(); break; case ID_MEMCARDPATH_B: DefaultMemcard[SLOT_B] = m_MemcardPath[SLOT_B]->GetPath().mb_str(); break; case NUMBER_OF_COLUMN: for( int i = COLUMN_GAMECODE; i < NUMBER_OF_COLUMN; i++) { m_MemcardList[SLOT_A]->column[i] = !m_MemcardList[SLOT_A]->column[i]; m_MemcardList[SLOT_B]->column[i] = !m_MemcardList[SLOT_B]->column[i]; } default: m_MemcardList[SLOT_A]->column[event.GetId()] = !m_MemcardList[SLOT_A]->column[event.GetId()]; m_MemcardList[SLOT_B]->column[event.GetId()] = !m_MemcardList[SLOT_B]->column[event.GetId()]; if (memoryCard[SLOT_A]) ReloadMemcard(m_MemcardPath[SLOT_A]->GetPath().mb_str(), SLOT_A); if (memoryCard[SLOT_B]) ReloadMemcard(m_MemcardPath[SLOT_B]->GetPath().mb_str(), SLOT_B); break; } } bool CMemcardManager::CopyDeleteSwitch(u32 error, int slot) { switch (error) { case GCS: SuccessAlert("File converted to .gci"); break; case SUCCESS: if (slot != -1) { memoryCard[slot]->FixChecksums(); if (!memoryCard[slot]->Save()) PanicAlert(E_SAVEFAILED); page[slot] = FIRSTPAGE; ReloadMemcard(m_MemcardPath[slot]->GetPath().mb_str(), slot); } break; case NOMEMCARD: PanicAlert("File is not recognized as a memcard"); break; case OPENFAIL: PanicAlert("File could not be opened\nor does not have a valid extension"); break; case OUTOFBLOCKS: if (slot == -1) { PanicAlert(E_UNK); break; } PanicAlert("Only %d blocks available", memoryCard[slot]->GetFreeBlocks()); break; case OUTOFDIRENTRIES: PanicAlert("No free dir index entries"); break; case LENGTHFAIL: PanicAlert("Imported file has invalid length"); break; case INVALIDFILESIZE: PanicAlert("The save you are trying to copy has an invalid file size"); break; case TITLEPRESENT: PanicAlert("Memcard already has a save for this title"); break; case SAVFAIL: PanicAlert("Imported file has sav extension\nbut does not have a correct header"); break; case GCSFAIL: PanicAlert("Imported file has gsc extension\nbut does not have a correct header"); break; case FAIL: if (slot == -1) { PanicAlert("Export Failed"); return false; } PanicAlert("Invalid bat.map or dir entry"); break; case WRITEFAIL: PanicAlert(E_SAVEFAILED); break; case DELETE_FAIL: PanicAlert("Order of files in the File Directory do not match the block order\n" "Right click and export all of the saves,\nand import the the saves to a new memcard\n"); break; default: PanicAlert(E_UNK); break; } return true; } void CMemcardManager::CopyDeleteClick(wxCommandEvent& event) { int index_A = m_MemcardList[SLOT_A]->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED); int index_B = m_MemcardList[SLOT_B]->GetNextItem(-1, wxLIST_NEXT_ALL, wxLIST_STATE_SELECTED); int slot = SLOT_B; int slot2 = SLOT_A; int index = index_B; std::string fileName2(""); if (index_A != wxNOT_FOUND && page[SLOT_A]) index_A += itemsPerPage * page[SLOT_A]; if (index_B != wxNOT_FOUND && page[SLOT_B]) index_B += itemsPerPage * page[SLOT_B]; switch (event.GetId()) { case ID_COPYFROM_B: slot = SLOT_A; slot2 = SLOT_B; case ID_COPYFROM_A: index = slot2 ? index_B : index_A; if ((index != wxNOT_FOUND)) { CopyDeleteSwitch(memoryCard[slot]->CopyFrom(*memoryCard[slot2], index), slot); } break; case ID_FIXCHECKSUM_A: slot = SLOT_A; case ID_FIXCHECKSUM_B: if (memoryCard[slot]->FixChecksums() && memoryCard[slot]->Save()) { SuccessAlert("The checksum was successfully fixed"); } else PanicAlert(E_SAVEFAILED); break; case ID_CONVERTTOGCI: fileName2 = "convert"; case ID_SAVEIMPORT_A: slot = SLOT_A; case ID_SAVEIMPORT_B: { wxString temp = wxFileSelector(_T("Select a save file to import"), (strcmp(DefaultIOPath.c_str(), "/Users/GC") == 0) ? wxString::FromAscii(""): wxString::FromAscii(DefaultIOPath.c_str()), wxEmptyString, wxEmptyString, wxString::Format ( _T("Gamecube save files(*.gci,*.gcs,*.sav)|*.gci;*.gcs;*.sav|" "Native GCI files (*.gci)|*.gci|" "MadCatz Gameshark files(*.gcs)|*.gcs|" "Datel MaxDrive/Pro files(*.sav)|*.sav"), wxFileSelectorDefaultWildcardStr, wxFileSelectorDefaultWildcardStr ), wxFD_OPEN | wxFD_FILE_MUST_EXIST); const char * fileName = temp.ToAscii(); if (!temp.empty() && !fileName2.empty()) { wxString temp2 = wxFileSelector(_T("Save GCI as.."), wxEmptyString, wxEmptyString, _T(".gci"), wxString::Format ( _T("GCI File(*.gci)|*.gci"), wxFileSelectorDefaultWildcardStr, wxFileSelectorDefaultWildcardStr ), wxFD_OVERWRITE_PROMPT|wxFD_SAVE); if (temp2.empty()) break; fileName2 = temp2.mb_str(); } if (temp.length() > 0) { CopyDeleteSwitch(memoryCard[slot]->ImportGci(fileName, fileName2), slot); } } break; case ID_SAVEEXPORT_A: slot=SLOT_A; index = index_A; case ID_SAVEEXPORT_B: if (index != wxNOT_FOUND) { char tempC[10 + DENTRY_STRLEN], tempC2[DENTRY_STRLEN]; memoryCard[slot]->DEntry_GameCode(index,tempC); memoryCard[slot]->DEntry_FileName(index,tempC2); sprintf(tempC, "%s_%s.gci", tempC, tempC2); wxString temp = wxFileSelector(_T("Export save as.."), wxString::FromAscii(DefaultIOPath.c_str()), wxString::FromAscii(tempC), _T(".gci"), wxString::Format ( _T("Native GCI files (*.gci)|*.gci|" "MadCatz Gameshark files(*.gcs)|*.gcs|" "Datel MaxDrive/Pro files(*.sav)|*.sav"), wxFileSelectorDefaultWildcardStr, wxFileSelectorDefaultWildcardStr ), wxFD_OVERWRITE_PROMPT|wxFD_SAVE); if (temp.length() > 0) { const char * fileName = temp.ToAscii(); if (!CopyDeleteSwitch(memoryCard[slot]->ExportGci(index, fileName, NULL), -1)) { File::Delete(temp.ToAscii()); } } } break; case ID_EXPORTALL_A: slot=SLOT_A; case ID_EXPORTALL_B: { std::string path1, path2, mpath; mpath = m_MemcardPath[slot]->GetPath().mb_str(); SplitPath(mpath, &path1, &path2, NULL); path1 += path2; File::CreateDir(path1.c_str()); if(PanicYesNo("Warning: This will overwrite any existing saves " "that are in the folder:\n%s\nand have the same name" " as a file on your memcard\nContinue?", path1.c_str())) for (int i = 0; i < DIRLEN; i++) { CopyDeleteSwitch(memoryCard[slot]->ExportGci(i, ".", &path1), -1); } break; } case ID_DELETE_A: slot = SLOT_A; index = index_A; case ID_DELETE_B: if (index != wxNOT_FOUND) { CopyDeleteSwitch(memoryCard[slot]->RemoveFile(index), slot); } break; } } bool CMemcardManager::ReloadMemcard(const char *fileName, int card) { wxString wxBlock; wxString wxFirstBlock; wxString wxLabel; wxString tString; int j; if (memoryCard[card]) delete memoryCard[card]; // TODO: add error checking and animate icons memoryCard[card] = new GCMemcard(fileName); if (memoryCard[card]->fail) return false; m_MemcardList[card]->Hide(); m_MemcardList[card]->ClearAll(); m_MemcardList[card]->InsertColumn(COLUMN_BANNER, _T("Banner")); m_MemcardList[card]->InsertColumn(COLUMN_TITLE, _T("Title")); m_MemcardList[card]->InsertColumn(COLUMN_COMMENT, _T("Comment")); m_MemcardList[card]->InsertColumn(COLUMN_ICON, _T("Icon")); m_MemcardList[card]->InsertColumn(COLUMN_BLOCKS, _T("Blocks")); m_MemcardList[card]->InsertColumn(COLUMN_FIRSTBLOCK, _T("First Block")); #ifdef DEBUG_MCM m_MemcardList[card]->InsertColumn(COLUMN_GAMECODE, _T("GameCode")); m_MemcardList[card]->InsertColumn(COLUMN_MAKERCODE, _T("MakerCode")); m_MemcardList[card]->InsertColumn(COLUMN_BIFLAGS, _T("BIFLAGS")); m_MemcardList[card]->InsertColumn(COLUMN_FILENAME, _T("FILENAME")); m_MemcardList[card]->InsertColumn(COLUMN_MODTIME, _T("MODTIME")); m_MemcardList[card]->InsertColumn(COLUMN_IMAGEADD, _T("IMAGEADD")); m_MemcardList[card]->InsertColumn(COLUMN_ICONFMT, _T("ICONFMT")); m_MemcardList[card]->InsertColumn(COLUMN_ANIMSPEED, _T("ANIMSPEED")); m_MemcardList[card]->InsertColumn(COLUMN_PERMISSIONS, _T("PERMISSIONS")); m_MemcardList[card]->InsertColumn(COLUMN_COPYCOUNTER, _T("COPYCOUNTER")); m_MemcardList[card]->InsertColumn(COLUMN_COMMENTSADDRESS, _T("COMMENTSADDRESS")); #endif wxImageList *list = m_MemcardList[card]->GetImageList(wxIMAGE_LIST_SMALL); list->RemoveAll(); int nFiles = memoryCard[card]->GetNumFiles(); int *images = new int[nFiles*2]; for (int i = 0;i < nFiles;i++) { static u32 pxdata[96*32]; static u8 animDelay[8]; static u32 animData[32*32*8]; int numFrames = memoryCard[card]->ReadAnimRGBA8(i,animData,animDelay); if (!memoryCard[card]->ReadBannerRGBA8(i,pxdata)) { memset(pxdata,0,96*32*4); if (numFrames>0) // Just use the first one { u32 *icdata = animData; for (int y=0;y<32;y++) { for (int x=0;x<32;x++) { pxdata[y*96+x+32] = icdata[y*32+x];// | 0xFF000000 } } } } wxBitmap map = wxBitmapFromMemoryRGBA((u8*)pxdata,96,32); images[i*2] = list->Add(map); if (numFrames>0) { memset(pxdata,0,96*32*4); int frames=3; if (numFramesAdd(icon); } } int pagesMax = 128; if (m_MemcardList[card]->usePages) pagesMax = (page[card] + 1) * itemsPerPage; for (j = page[card] * itemsPerPage;(j < nFiles) && (j < pagesMax); j++) { char title[DENTRY_STRLEN]; char comment[DENTRY_STRLEN]; u16 blocks; u16 firstblock; int index = m_MemcardList[card]->InsertItem(j, wxEmptyString); m_MemcardList[card]->SetItem(index, COLUMN_BANNER, wxEmptyString); if (!memoryCard[card]->DEntry_Comment1(j, title)) title[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_TITLE, wxString::FromAscii(title)); if (!memoryCard[card]->DEntry_Comment2(j, comment)) comment[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_COMMENT, wxString::FromAscii(comment)); blocks = memoryCard[card]->DEntry_BlockCount(j); if (blocks == 0xFFFF) blocks = 0; wxBlock.Printf(wxT("%10d"), blocks); m_MemcardList[card]->SetItem(index,COLUMN_BLOCKS, wxBlock); firstblock = memoryCard[card]->DEntry_FirstBlock(j); if (firstblock == 0xFFFF) firstblock = 3; // to make firstblock -1 wxFirstBlock.Printf(wxT("%15d"), firstblock-4); m_MemcardList[card]->SetItem(index,COLUMN_FIRSTBLOCK, wxFirstBlock); m_MemcardList[card]->SetItem(index, COLUMN_ICON, wxEmptyString); if (images[j] >= 0) { m_MemcardList[card]->SetItemImage(index, images[j*2]); m_MemcardList[card]->SetItemColumnImage(index, COLUMN_ICON, images[j*2+1]); } #ifdef DEBUG_MCM char gC[5]; if (!memoryCard[card]->DEntry_GameCode(j, gC)) gC[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_GAMECODE, wxString::FromAscii(gC)); char mC[3]; if (!memoryCard[card]->DEntry_Markercode(j, mC)) mC[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_MAKERCODE, wxString::FromAscii(mC)); char bI[9]; if (!memoryCard[card]->DEntry_BIFlags(j, bI)) bI[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_BIFLAGS, wxString::FromAscii(bI)); char fN[32]; if (!memoryCard[card]->DEntry_FileName(j, fN)) fN[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_FILENAME, wxString::FromAscii(fN)); tString.Printf(wxT("%04X"), memoryCard[card]->DEntry_ModTime(j)); m_MemcardList[card]->SetItem(index, COLUMN_MODTIME, tString); tString.Printf(wxT("%04X"), memoryCard[card]->DEntry_ImageOffset(j)); m_MemcardList[card]->SetItem(index, COLUMN_IMAGEADD, tString); char iF[17]; if (!memoryCard[card]->DEntry_IconFmt(j, iF)) iF[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_ICONFMT, wxString::FromAscii(iF)); tString.Printf(wxT("%02X"), memoryCard[card]->DEntry_AnimSpeed(j)); m_MemcardList[card]->SetItem(index, COLUMN_ANIMSPEED, tString); char per[40]; if (!memoryCard[card]->DEntry_Permissions(j, per)) per[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_PERMISSIONS, wxString::FromAscii(per)); tString.Printf(wxT("%0X"), memoryCard[card]->DEntry_CopyCounter(j)); m_MemcardList[card]->SetItem(index, COLUMN_COPYCOUNTER, tString); tString.Printf(wxT("%04X"), memoryCard[card]->DEntry_CommentsAddress(j)); m_MemcardList[card]->SetItem(index, COLUMN_COMMENTSADDRESS, tString); #endif } if (m_MemcardList[card]->usePages) { if (nFiles <= itemsPerPage) { m_PrevPage[card]->Disable(); m_MemcardList[card]->prevPage = false; } if (j == nFiles) { m_NextPage[card]->Disable(); m_MemcardList[card]->nextPage = false; } else { m_NextPage[card]->Enable(); m_MemcardList[card]->nextPage = true; } } delete[] images; // Automatic column width and then show the list for (int i = 0; i < NUMBER_OF_COLUMN; i++) { if (m_MemcardList[card]->column[i]) m_MemcardList[card]->SetColumnWidth(i, wxLIST_AUTOSIZE); else m_MemcardList[card]->SetColumnWidth(i, 0); } m_MemcardList[card]->Show(); wxLabel.Printf(wxT("%d Free Blocks; %d Free Dir Entries"), memoryCard[card]->GetFreeBlocks(), DIRLEN - nFiles); t_Status[card]->SetLabel(wxLabel); #ifdef DEBUG_MCM if(MemcardManagerDebug == NULL) { MemcardManagerDebug = new CMemcardManagerDebug((wxFrame *)NULL, wxDefaultPosition, wxSize(950, 400)); } if (MemcardManagerDebug != NULL) { MemcardManagerDebug->Show(); MemcardManagerDebug->updatePanels(memoryCard, card); } #endif return true; } void CMemcardManager::CMemcardListCtrl::OnRightClick(wxMouseEvent& event) { int flags; long item = HitTest(event.GetPosition(), flags); wxMenu popupMenu; if (item != wxNOT_FOUND) { if (GetItemState(item, wxLIST_STATE_SELECTED) != wxLIST_STATE_SELECTED) { SetItemState(item, wxLIST_STATE_SELECTED, wxLIST_STATE_SELECTED ); } SetItemState(item, wxLIST_STATE_FOCUSED, wxLIST_STATE_FOCUSED); int slot = event.GetId() - ID_MEMCARDLIST_A; popupMenu.Append(ID_COPYFROM_A + slot, wxString::Format(wxT("Copy to Memcard %c"), 'B' - slot)); popupMenu.Append(ID_DELETE_A + slot, wxT("Delete Save")); popupMenu.Append(ID_SAVEIMPORT_A + slot, wxT("Import Save")); popupMenu.Append(ID_SAVEEXPORT_A + slot, wxT("Export Save")); popupMenu.Append(ID_EXPORTALL_A + slot, wxT("Export all saves")); if (!twoCardsLoaded) popupMenu.FindItem(ID_COPYFROM_A + slot)->Enable(false); popupMenu.AppendSeparator(); popupMenu.Append(ID_FIXCHECKSUM_A + slot, wxT("Fix Checksums")); popupMenu.Append(ID_PREVPAGE_A + slot, wxT("Previous Page")); popupMenu.Append(ID_NEXTPAGE_A + slot, wxT("Next Page")); popupMenu.Append(ID_MEMCARDPATH_A + slot, wxString::Format(wxT("Set as default Memcard %c"), 'A' + slot)); popupMenu.AppendCheckItem(ID_USEPAGES, wxT("Enable pages")); if (!prevPage || !usePages) popupMenu.FindItem(ID_PREVPAGE_A + slot)->Enable(false); if (!nextPage || !usePages) popupMenu.FindItem(ID_NEXTPAGE_A + slot)->Enable(false); if(usePages) popupMenu.FindItem(ID_USEPAGES)->Check(); popupMenu.AppendSeparator(); popupMenu.AppendCheckItem(COLUMN_BANNER, wxT("Show save banner")); popupMenu.AppendCheckItem(COLUMN_TITLE, wxT("Show save title")); popupMenu.AppendCheckItem(COLUMN_COMMENT, wxT("Show save comment")); popupMenu.AppendCheckItem(COLUMN_ICON, wxT("Show save icon")); popupMenu.AppendCheckItem(COLUMN_BLOCKS, wxT("Show save blocks")); for (int i = COLUMN_BANNER; i <= COLUMN_BLOCKS; i++) { if (column[i]) popupMenu.FindItem(i)->Check(); } #ifdef DEBUG_MCM popupMenu.AppendCheckItem(NUMBER_OF_COLUMN, wxT("Debug Memcard")); if (column[NUMBER_OF_COLUMN]) popupMenu.FindItem(NUMBER_OF_COLUMN)->Check(); #endif } PopupMenu(&popupMenu); }