// 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 "Globals.h" #include "MemcardManager.h" #include "wx/mstream.h" 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;yAssignImageList(new wxImageList(96,32),wxIMAGE_LIST_SMALL); m_MemcardList[SLOT_B]->AssignImageList(new wxImageList(96,32),wxIMAGE_LIST_SMALL); t_Status_A = new wxStaticText(this, 0, wxEmptyString, wxDefaultPosition,wxDefaultSize, 0, wxEmptyString); t_Status_B = new wxStaticText(this, 0, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0, wxEmptyString); // buttons m_CopyTo_A = new wxButton(this, ID_COPYTO_A, wxT("<-Copy<-"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_CopyTo_B = new wxButton(this, ID_COPYTO_B, wxT("->Copy->"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_FixChecksum_A = new wxButton(this, ID_FIXCHECKSUM_A, wxT("<-Fix Checksum"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_FixChecksum_B = new wxButton(this, ID_FIXCHECKSUM_B, wxT("Fix Checksum->"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_SaveImport_A = new wxButton(this, ID_SAVEIMPORT_A, wxT("<-Import GCI"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_SaveImport_B = new wxButton(this, ID_SAVEIMPORT_B, wxT("Import GCI->"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_SaveExport_A = new wxButton(this, ID_SAVEEXPORT_A, wxT("<-Export GCI"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_SaveExport_B = new wxButton(this, ID_SAVEEXPORT_B, wxT("Export GCI->"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_ConvertToGci = new wxButton(this, ID_CONVERTTOGCI, wxT("Convert to GCI"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_Delete_A = new wxButton(this, ID_DELETE_A, wxT("<-Delete"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_Delete_B = new wxButton(this, ID_DELETE_B, wxT("Delete->"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_PrevPage_A = new wxButton(this, ID_PREVPAGE_A, wxT("Prev Page"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_PrevPage_B = new wxButton(this, ID_PREVPAGE_B, wxT("Prev Page"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_NextPage_A = new wxButton(this, ID_NEXTPAGE_A, wxT("Next Page"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); m_NextPage_B = new wxButton(this, ID_NEXTPAGE_B, wxT("Next Page"), wxDefaultPosition, wxDefaultSize, 0, wxDefaultValidator); // Sizers that double as wxStaticBoxes sMemcard_A = new wxStaticBoxSizer(wxVERTICAL, this, wxT("Memory Card A")); sMemcard_B = new wxStaticBoxSizer(wxVERTICAL, this, wxT("Memory Card B")); // mmmm sizer goodness wxBoxSizer* sButtons; sButtons = new wxBoxSizer(wxVERTICAL); sButtons->AddStretchSpacer(2); sButtons->Add(m_CopyTo_A, 0, wxEXPAND, 5); sButtons->Add(m_CopyTo_B, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_FixChecksum_A, 0, wxEXPAND, 5); sButtons->Add(m_FixChecksum_B, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_SaveImport_A, 0, wxEXPAND, 5); sButtons->Add(m_SaveExport_A, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_ConvertToGci, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_SaveImport_B, 0, wxEXPAND, 5); sButtons->Add(m_SaveExport_B, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sButtons->Add(m_Delete_A, 0, wxEXPAND, 5); sButtons->Add(m_Delete_B, 0, wxEXPAND, 5); sButtons->AddStretchSpacer(1); sPages_A = new wxBoxSizer(wxHORIZONTAL); sPages_B = new wxBoxSizer(wxHORIZONTAL); sPages_A->Add(m_PrevPage_A, 0, wxEXPAND|wxALL, 1); sPages_A->Add(t_Status_A,0, wxEXPAND|wxALL, 5); sPages_A->Add(0, 0, 1, wxEXPAND|wxALL, 0); sPages_A->Add(m_NextPage_A, 0, wxEXPAND|wxALL, 1); sPages_B->Add(m_PrevPage_B, 0, wxEXPAND|wxALL, 1); sPages_B->Add(t_Status_B, 0, wxEXPAND|wxALL, 5); sPages_B->Add(0, 0, 1, wxEXPAND|wxALL, 0); sPages_B->Add(m_NextPage_B, 0, wxEXPAND|wxALL, 1); sMemcard_A->Add(m_MemcardPath_A, 0, wxEXPAND|wxALL, 5); sMemcard_A->Add(m_MemcardList[SLOT_A], 1, wxEXPAND|wxALL, 5); sMemcard_A->Add(sPages_A, 0, wxEXPAND|wxALL, 1); sMemcard_B->Add(m_MemcardPath_B, 0, wxEXPAND|wxALL, 5); sMemcard_B->Add(m_MemcardList[SLOT_B], 1, wxEXPAND|wxALL, 5); sMemcard_B->Add(sPages_B, 0, wxEXPAND|wxALL, 1); sMain = new wxBoxSizer(wxHORIZONTAL); sMain->Add(sMemcard_A, 1, wxEXPAND|wxALL, 5); sMain->Add(sButtons, 0, wxEXPAND, 0); sMain->Add(sMemcard_B, 1, wxEXPAND|wxALL, 5); this->SetSizer(sMain); sMain->SetSizeHints(this); Fit(); m_PrevPage_A->Disable(); m_NextPage_A->Disable(); m_PrevPage_B->Disable(); m_NextPage_B->Disable(); m_CopyTo_A->Disable(); m_CopyTo_B->Disable(); m_FixChecksum_A->Disable(); m_FixChecksum_B->Disable(); m_SaveImport_A->Disable(); m_SaveExport_A->Disable(); m_SaveImport_B->Disable(); m_SaveExport_B->Disable(); m_Delete_A->Disable(); m_Delete_B->Disable(); } void CMemcardManager::OnClose(wxCloseEvent& WXUNUSED (event)) { Destroy(); } void CMemcardManager::OnPathChange(wxFileDirPickerEvent& event) { switch (event.GetId()) { case ID_MEMCARDPATH_A: pageA = FIRSTPAGE; if (m_MemcardList[SLOT_A]->usePages && m_PrevPage_A->IsEnabled()) { m_PrevPage_A->Disable(); m_MemcardList[SLOT_A]->prevPage = false; } if (!strcasecmp(m_MemcardPath_A->GetPath().mb_str(), m_MemcardPath_B->GetPath().mb_str())) { wxMessageBox(wxT("Memcard already opened"), wxT("Error"), wxOK|wxICON_ERROR); } else if (ReloadMemcard(event.GetPath().mb_str(), SLOT_A, pageA)) { m_MemcardList[SLOT_B]->twoCardsLoaded = true; m_FixChecksum_A->Enable(); m_SaveImport_A->Enable(); m_SaveExport_A->Enable(); m_Delete_A->Enable(); break; } m_MemcardList[SLOT_B]->twoCardsLoaded = false; m_MemcardPath_A->SetPath(wxEmptyString); m_MemcardList[SLOT_A]->ClearAll(); t_Status_A->SetLabel(wxEmptyString); m_FixChecksum_A->Disable(); m_SaveImport_A->Disable(); m_SaveExport_A->Disable(); m_Delete_A->Disable(); if (m_MemcardList[SLOT_A]->usePages) { m_PrevPage_A->Disable(); m_NextPage_A->Disable(); } break; case ID_MEMCARDPATH_B: pageB = FIRSTPAGE; if (m_MemcardList[SLOT_B]->usePages && m_PrevPage_B->IsEnabled()) { m_PrevPage_B->Disable(); m_MemcardList[SLOT_B]->prevPage = false; } if (!strcasecmp(m_MemcardPath_A->GetPath().mb_str(), m_MemcardPath_B->GetPath().mb_str())) { wxMessageBox(wxT("Memcard already opened"), wxT("Error"), wxOK|wxICON_ERROR); } else if (ReloadMemcard(event.GetPath().mb_str(), SLOT_B, pageB)) { m_MemcardList[SLOT_A]->twoCardsLoaded = true; m_FixChecksum_B->Enable(); m_SaveImport_B->Enable(); m_SaveExport_B->Enable(); m_Delete_B->Enable(); break; } m_MemcardList[SLOT_A]->twoCardsLoaded = false; m_MemcardPath_B->SetPath(wxEmptyString); m_MemcardList[SLOT_B]->ClearAll(); t_Status_B->SetLabel(wxEmptyString); m_FixChecksum_B->Disable(); m_SaveImport_B->Disable(); m_SaveExport_B->Disable(); m_Delete_B->Disable(); if (m_MemcardList[SLOT_B]->usePages) { m_PrevPage_B->Disable(); m_NextPage_B->Disable(); } break; } if (m_Delete_B->IsEnabled() && m_Delete_A->IsEnabled()) { m_CopyTo_A->Enable(); m_CopyTo_B->Enable(); } else { m_CopyTo_A->Disable(); m_CopyTo_B->Disable(); } } void CMemcardManager::OnPageChange(wxCommandEvent& event) { switch (event.GetId()) { case ID_NEXTPAGE_A: if (!m_PrevPage_A->IsEnabled()) { m_PrevPage_A->Enable(); m_MemcardList[SLOT_A]->prevPage = true; } pageA++; if (pageA == maxPages) { m_NextPage_A->Disable(); m_MemcardList[SLOT_A]->nextPage = false; } ReloadMemcard(m_MemcardPath_A->GetPath().mb_str(), SLOT_A, pageA); break; case ID_NEXTPAGE_B: if (!m_PrevPage_B->IsEnabled()) { m_PrevPage_B->Enable(); m_MemcardList[SLOT_B]->prevPage = true; } pageB++; if (pageB == maxPages) { m_NextPage_B->Disable(); m_MemcardList[SLOT_B]->nextPage = false; } ReloadMemcard(m_MemcardPath_B->GetPath().mb_str(), SLOT_B, pageB); break; case ID_PREVPAGE_A: if (!m_NextPage_A->IsEnabled()) { m_NextPage_A->Enable(); m_MemcardList[SLOT_A]->nextPage = true; } pageA--; if (!pageA) { m_PrevPage_A->Disable(); m_MemcardList[SLOT_A]->prevPage = false; } ReloadMemcard(m_MemcardPath_A->GetPath().mb_str(), SLOT_A, pageA); break; case ID_PREVPAGE_B: if (!m_NextPage_B->IsEnabled()) { m_NextPage_B->Enable(); m_MemcardList[SLOT_B]->nextPage = true; } pageB--; if (!pageB) { m_PrevPage_B->Disable(); m_MemcardList[SLOT_B]->prevPage = false; } ReloadMemcard(m_MemcardPath_B->GetPath().mb_str(), SLOT_B, pageB); break; } } void CMemcardManager::OnMenuChange(wxCommandEvent& event) { if (event.GetId() == 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_A->Disable(); m_PrevPage_B->Disable(); m_NextPage_A->Disable(); m_NextPage_B->Disable(); pageA = pageB = FIRSTPAGE; } } else { 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_A->GetPath().mb_str(), SLOT_A, pageA); if (memoryCard[SLOT_B]) ReloadMemcard(m_MemcardPath_B->GetPath().mb_str(), SLOT_B, pageB); } 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 index = index_B; std::string fileName2(""); wxString blocksOpen; if (index_A != wxNOT_FOUND && pageA) index_A += itemsPerPage * pageA; if (index_B != wxNOT_FOUND && pageB) index_B += itemsPerPage * pageB; switch (event.GetId()) { case ID_COPYTO_A: if ((index_B != wxNOT_FOUND) && (memoryCard[0] != NULL)) { switch (memoryCard[SLOT_A]->CopyFrom(*memoryCard[SLOT_B], index_B)) { case FAIL: wxMessageBox(wxT("Invalid bat.map or dir entry"), wxT("Failure"), wxOK); break; case NOMEMCARD: wxMessageBox(wxT("File is not recognized as a memcard"), wxT("Failure"), wxOK); break; case SUCCESS: memoryCard[SLOT_A]->Save(); ReloadMemcard(m_MemcardPath_A->GetPath().mb_str(), SLOT_A, FIRSTPAGE); break; } } else { wxMessageBox(wxT("You have not selected a save to copy"), wxT("Error"), wxOK|wxICON_ERROR); } break; case ID_COPYTO_B: if ((index_A != wxNOT_FOUND) && (memoryCard[SLOT_B] != NULL)) { switch (memoryCard[SLOT_B]->CopyFrom(*memoryCard[SLOT_A], index_A)) { case FAIL: wxMessageBox(wxT("Invalid bat.map or dir entry"), wxT("Error"), wxOK|wxICON_ERROR); break; case NOMEMCARD: wxMessageBox(wxT("File is not recognized as a memcard"), wxT("Error"), wxOK|wxICON_ERROR); break; case SUCCESS: memoryCard[SLOT_B]->Save(); ReloadMemcard(m_MemcardPath_B->GetPath().mb_str(), SLOT_B, FIRSTPAGE); break; } } else { wxMessageBox(wxT("You have not selected a save to copy"), wxT("Error"), wxOK|wxICON_ERROR); } break; case ID_FIXCHECKSUM_A: slot = SLOT_A; case ID_FIXCHECKSUM_B: if (memoryCard[slot] != NULL) { // Fix checksums and save the changes memoryCard[slot]->FixChecksums() ? wxMessageBox(wxT("The checksum was successfully fixed"), wxT("Success"), wxOK) : wxMessageBox(wxT("The checksum could not be successfully fixed"), wxT("Error"), wxOK|wxICON_ERROR); memoryCard[slot]->Save(); } else { wxMessageBox(wxT("memorycard is not loaded"), wxT("Error"), wxOK|wxICON_ERROR); } break; case ID_CONVERTTOGCI: fileName2 = "convert"; case ID_SAVEIMPORT_A: slot = SLOT_A; case ID_SAVEIMPORT_B: if (memoryCard[slot] != NULL || !fileName2.empty()) { wxString temp = wxFileSelector(_T("Select a save file to import"), wxEmptyString, 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) { switch(memoryCard[slot]->ImportGci(fileName, fileName2)) { case LENGTHFAIL: wxMessageBox(wxT("Imported file has invalid length"), wxT("Error"), wxOK|wxICON_ERROR); break; case GCSFAIL: wxMessageBox(wxT("Imported file has gsc extension\nbut" " does not have a correct header"), wxT("Error"), wxOK|wxICON_ERROR); break; case SAVFAIL: wxMessageBox(wxT("Imported file has sav extension\nbut" " does not have a correct header"), wxT("Error"), wxOK|wxICON_ERROR); break; case OPENFAIL: wxMessageBox(wxT("Imported file could not be opened\nor" " does not have a valid extension"), wxT("Error"), wxOK|wxICON_ERROR); break; case GCS: wxMessageBox(wxT("File converted to .gci"), wxT("Success"), wxOK); break; case OUTOFBLOCKS: blocksOpen.Printf(wxT("Only %d blocks available"), memoryCard[slot]->GetFreeBlocks()); wxMessageBox(blocksOpen, wxT("Error"), wxOK|wxICON_ERROR); break; case OUTOFDIRENTRIES: wxMessageBox(wxT("No free dir index entries"), wxT("Error"), wxOK|wxICON_ERROR); break; case NOMEMCARD: wxMessageBox(wxT("File is not recognized as a memcard"), wxT("Error"), wxOK|wxICON_ERROR); break; case TITLEPRESENT: wxMessageBox(wxT("Memcard already has a save for this title"), wxT("Error"), wxOK|wxICON_ERROR); break; case FAIL: wxMessageBox(wxT("Invalid bat.map or dir entry"), wxT("Error"), wxOK|wxICON_ERROR); break; default: memoryCard[slot]->Save(); slot == SLOT_B ? ReloadMemcard(m_MemcardPath_B->GetPath().mb_str(), SLOT_B, FIRSTPAGE) : ReloadMemcard(m_MemcardPath_A->GetPath().mb_str(), SLOT_A, FIRSTPAGE); break; } } } else { wxMessageBox(wxT("Memory card is not loaded"), wxT("Error"), wxOK|wxICON_ERROR); } break; case ID_SAVEEXPORT_A: slot=SLOT_A; index = index_A; case ID_SAVEEXPORT_B: if (index != wxNOT_FOUND) { wxString temp = wxFileSelector(_T("Save GCI as.."), wxEmptyString, wxEmptyString, _T(".gci"), wxString::Format ( _T("GCI File(*.gci)|*.gci"), wxFileSelectorDefaultWildcardStr, wxFileSelectorDefaultWildcardStr ), wxFD_OVERWRITE_PROMPT|wxFD_SAVE); const char * fileName = temp.ToAscii(); if (temp.length() > 0) { switch (memoryCard[slot]->ExportGci(index, fileName)) { case NOMEMCARD: wxMessageBox(wxT("File is not recognized as a memcard"), wxT("Error"), wxOK|wxICON_ERROR); break; case NOFILE: wxMessageBox(wxT("Could not open gci for writing"), wxT("Error"), wxOK|wxICON_ERROR); break; case FAIL: //TODO: delete file if fails wxMessageBox(wxT("Invalid bat.map or dir entry"), wxT("Error"), wxOK|wxICON_ERROR); break; default: break; } } } else { wxMessageBox(wxT("You have not selected a save to export"), wxT("Error"), wxOK|wxICON_ERROR); } break; case ID_DELETE_A: slot=SLOT_A; index = index_A; case ID_DELETE_B: if (index != wxNOT_FOUND) { switch (memoryCard[slot]->RemoveFile(index)) { case NOMEMCARD: wxMessageBox(wxT("File is not recognized as a memcard"), wxT("Error"), wxOK|wxICON_ERROR); break; case FAIL: wxMessageBox(wxT("Invalid bat.map or dir entry"), wxT("Error"), wxOK|wxICON_ERROR); break; case SUCCESS: memoryCard[slot]->Save(); slot == SLOT_B ? ReloadMemcard(m_MemcardPath_B->GetPath().mb_str(), SLOT_B, FIRSTPAGE) : ReloadMemcard(m_MemcardPath_A->GetPath().mb_str(), SLOT_A, FIRSTPAGE); break; } } else { wxMessageBox(wxT("You have not selected a save to delete"), wxT("Error"), wxOK|wxICON_ERROR); } break; } } bool CMemcardManager::ReloadMemcard(const char *fileName, int card, int page) { wxString wxBlock; wxString wxFirstBlock; wxString wxLabel; int j; if (memoryCard[card]) delete memoryCard[card]; // TODO: add error checking and animate icons memoryCard[card] = new GCMemcard(fileName); if (ReadError(memoryCard[card])) 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")); 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 + 1) * itemsPerPage; for (j = page * itemsPerPage;(j < nFiles) && (j < pagesMax);j++) { char title[32]; char comment[32]; u16 blocks; u16 firstblock; int index = m_MemcardList[card]->InsertItem(j, wxEmptyString); m_MemcardList[card]->SetItem(index, COLUMN_BANNER, wxEmptyString); if (!memoryCard[card]->GetComment1(j, title)) title[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_TITLE, wxString::FromAscii(title)); if (!memoryCard[card]->GetComment2(j, comment)) comment[0]=0; m_MemcardList[card]->SetItem(index, COLUMN_COMMENT, wxString::FromAscii(comment)); blocks = memoryCard[card]->GetFileSize(j); if (blocks == 0xFFFF) blocks = 0; wxBlock.Printf(wxT("%10d"), blocks); m_MemcardList[card]->SetItem(index,COLUMN_BLOCKS, wxBlock); firstblock = memoryCard[card]->GetFirstBlock(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]); } } if (m_MemcardList[card]->usePages) { if ((j == nFiles)) { if (card) { m_NextPage_B->Disable(); m_MemcardList[SLOT_B]->nextPage = false; } else { m_NextPage_A->Disable(); m_MemcardList[SLOT_A]->nextPage = false; } } else { if (card) { m_NextPage_B->Enable(); m_MemcardList[SLOT_B]->nextPage = true; } else { m_NextPage_A->Enable(); m_MemcardList[SLOT_A]->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(), 127 - nFiles); card ? t_Status_B->SetLabel(wxLabel) : t_Status_A->SetLabel(wxLabel); return true; } bool CMemcardManager::ReadError(GCMemcard *memcard) { if (!memcard->fail[0]) return false; wxString wxBlock; if (memcard->fail[HDR_READ_ERROR]) wxMessageBox(wxT("Failed to read header correctly\n(0x0000-0x1FFF)"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[DIR_READ_ERROR]) wxMessageBox(wxT("Failed to read directory correctly\n(0x2000-0x3FFF)"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[DIR_BAK_READ_ERROR]) wxMessageBox(wxT("Failed to read directory backup correctly\n(0x4000-0x5FFF)"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[BAT_READ_ERROR]) wxMessageBox(wxT("Failed to read block allocation table correctly\n(0x6000-0x7FFF)"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[BAT_BAK_READ_ERROR]) wxMessageBox(wxT("Failed to read block allocation table backup correctly\n(0x8000-0x9FFF)"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[HDR_CSUM_FAIL]) wxMessageBox(wxT("Header checksum failed"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[DIR_CSUM_FAIL]) { wxMessageBox(wxT("Directory checksum failed"), wxT("Error"), wxOK|wxICON_ERROR); wxMessageBox(wxT("Directory backup checksum failed"), wxT("Error"), wxOK|wxICON_ERROR); } if (memcard->fail[BAT_CSUM_FAIL]) wxMessageBox(wxT("Block Allocation Table checksum failed"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[DATA_READ_FAIL]) wxMessageBox(wxT("Failed to read save data\n(0xA000-)\nMemcard may be truncated"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[HDR_SIZE_FFFF]) wxMessageBox(wxT("Memcard failed to load\n Card size is invalid"), wxT("Error"), wxOK|wxICON_ERROR); if (memcard->fail[NOTRAWORGCP]) wxMessageBox(wxT("File does not have a valid extension (.raw/.gcp)"), wxT("Error"), wxOK|wxICON_ERROR); return true; } void CMemcardManager::CMemcardListCtrl::OnRightClick(wxMouseEvent& event) { int flags; long item = HitTest(event.GetPosition(), flags); 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); wxMenu popupMenu; if (event.GetId() == ID_MEMCARDLIST_A) { popupMenu.Append(ID_COPYTO_B, wxT("Copy to Memcard B")); popupMenu.Append(ID_DELETE_A, wxT("Delete Save")); popupMenu.Append(ID_SAVEIMPORT_A, wxT("Import Save")); popupMenu.Append(ID_SAVEEXPORT_A, wxT("Export Save")); if (!twoCardsLoaded) popupMenu.FindItem(ID_COPYTO_B)->Enable(false); popupMenu.AppendSeparator(); popupMenu.Append(ID_FIXCHECKSUM_A, wxT("Fix Checksum")); popupMenu.Append(ID_PREVPAGE_A, wxT("Previous Page")); popupMenu.Append(ID_NEXTPAGE_A, wxT("Next Page")); if (!prevPage || !usePages) popupMenu.FindItem(ID_PREVPAGE_A)->Enable(false); if (!nextPage || !usePages) popupMenu.FindItem(ID_NEXTPAGE_A)->Enable(false); } else if (event.GetId() == ID_MEMCARDLIST_B) { popupMenu.Append(ID_COPYTO_A, wxT("Copy to Memcard A")); popupMenu.Append(ID_DELETE_B, wxT("Delete Save")); popupMenu.Append(ID_SAVEIMPORT_B, wxT("Import Save")); popupMenu.Append(ID_SAVEEXPORT_B, wxT("Export Save")); if (!twoCardsLoaded) popupMenu.FindItem(ID_COPYTO_A)->Enable(false); popupMenu.AppendSeparator(); popupMenu.Append(ID_FIXCHECKSUM_B, wxT("Fix Checksum")); popupMenu.Append(ID_PREVPAGE_B, wxT("Previous Page")); popupMenu.Append(ID_NEXTPAGE_B, wxT("Next Page")); if (!prevPage || !usePages) popupMenu.FindItem(ID_PREVPAGE_B)->Enable(false); if (!nextPage || !usePages) popupMenu.FindItem(ID_NEXTPAGE_B)->Enable(false); } popupMenu.AppendCheckItem(COLUMN_BANNER, wxT("Show save banner")); if(column[COLUMN_BANNER]) popupMenu.FindItem(COLUMN_BANNER)->Check(); popupMenu.AppendCheckItem(COLUMN_TITLE, wxT("Show save title")); if(column[COLUMN_TITLE]) popupMenu.FindItem(COLUMN_TITLE)->Check(); popupMenu.AppendCheckItem(COLUMN_COMMENT, wxT("Show save comment")); if(column[COLUMN_COMMENT]) popupMenu.FindItem(COLUMN_COMMENT)->Check(); popupMenu.AppendCheckItem(COLUMN_ICON, wxT("Show save icon")); if(column[COLUMN_ICON]) popupMenu.FindItem(COLUMN_ICON)->Check(); popupMenu.AppendCheckItem(COLUMN_BLOCKS, wxT("Show save blocks")); if(column[COLUMN_BLOCKS]) popupMenu.FindItem(COLUMN_BLOCKS)->Check(); popupMenu.AppendCheckItem(COLUMN_FIRSTBLOCK, wxT("Show save first block")); if(column[COLUMN_FIRSTBLOCK]) popupMenu.FindItem(COLUMN_FIRSTBLOCK)->Check(); popupMenu.AppendCheckItem(ID_USEPAGES, wxT("Enable pages")); if(usePages) popupMenu.FindItem(ID_USEPAGES)->Check(); PopupMenu(&popupMenu); } }