// Copyright 2008 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include "DolphinWX/MemcardManager.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "Common/CommonTypes.h" #include "Common/FileUtil.h" #include "Common/IniFile.h" #include "Common/Logging/Log.h" #include "Common/MsgHandler.h" #include "Common/StringUtil.h" #include "Core/HW/GCMemcard.h" #include "DolphinWX/WxUtils.h" #define FIRSTPAGE 0 #define ARROWS slot ? "" : ARROW[slot], slot ? ARROW[slot] : "" static wxImage wxImageFromMemoryRGBA(const u32* data, u32 width, u32 height) { wxImage image(width, height, false); image.InitAlpha(); u8* rgb = image.GetData(); u8* alpha = image.GetAlpha(); for (u32 y = 0; y < height; ++y) { for (u32 x = 0; x < width; ++x) { u32 pixel = data[y * width + x]; *rgb++ = (pixel & 0x00FF0000) >> 16; // Red *rgb++ = (pixel & 0x0000FF00) >> 8; // Green *rgb++ = (pixel & 0x000000FF) >> 0; // Blue *alpha++ = (pixel & 0xFF000000) >> 24; } } return image; } BEGIN_EVENT_TABLE(CMemcardManager, wxDialog) EVT_BUTTON(ID_COPYFROM_A, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_COPYFROM_B, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_DELETE_A, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_DELETE_B, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_SAVEIMPORT_B, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_SAVEEXPORT_B, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_SAVEIMPORT_A, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_SAVEEXPORT_A, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_CONVERTTOGCI, CMemcardManager::CopyDeleteClick) EVT_BUTTON(ID_PREVPAGE_A, CMemcardManager::OnPageChange) EVT_BUTTON(ID_NEXTPAGE_A, CMemcardManager::OnPageChange) EVT_BUTTON(ID_PREVPAGE_B, CMemcardManager::OnPageChange) EVT_BUTTON(ID_NEXTPAGE_B, CMemcardManager::OnPageChange) EVT_FILEPICKER_CHANGED(ID_MEMCARDPATH_A, CMemcardManager::OnPathChange) EVT_FILEPICKER_CHANGED(ID_MEMCARDPATH_B, CMemcardManager::OnPathChange) EVT_MENU_RANGE(ID_MEMCARDPATH_A, ID_USEPAGES, CMemcardManager::OnMenuChange) EVT_MENU_RANGE(ID_COPYFROM_A, ID_CONVERTTOGCI, CMemcardManager::CopyDeleteClick) EVT_MENU_RANGE(ID_NEXTPAGE_A, ID_PREVPAGE_B, CMemcardManager::OnPageChange) EVT_MENU_RANGE(COLUMN_BANNER, NUMBER_OF_COLUMN, CMemcardManager::OnMenuChange) END_EVENT_TABLE() CMemcardManager::CMemcardManager(wxWindow* parent) : wxDialog(parent, wxID_ANY, _("Memory Card Manager"), wxDefaultPosition, wxDefaultSize, wxCAPTION | wxSYSTEM_MENU | wxDIALOG_NO_PARENT | wxCLOSE_BOX | wxRESIZE_BORDER | wxMAXIMIZE_BOX), m_image_list_size(FromDIP(wxSize(96, 32))) { memoryCard[SLOT_A] = nullptr; memoryCard[SLOT_B] = nullptr; mcmSettings.twoCardsLoaded = false; if (!LoadSettings()) { itemsPerPage = 16; mcmSettings.usePages = true; for (int i = COLUMN_BANNER; i < NUMBER_OF_COLUMN; i++) { mcmSettings.column[i] = (i <= COLUMN_FIRSTBLOCK) ? true : false; } } maxPages = (128 / itemsPerPage) - 1; CreateGUIControls(); } CMemcardManager::~CMemcardManager() { SaveSettings(); } bool CMemcardManager::LoadSettings() { if (MemcardManagerIni.Load(File::GetUserPath(F_DOLPHINCONFIG_IDX))) { iniMemcardSection = MemcardManagerIni.GetOrCreateSection("MemcardManager"); iniMemcardSection->Get("Items per page", &itemsPerPage, 16); iniMemcardSection->Get("DefaultMemcardA", &(DefaultMemcard[SLOT_A]), ""); iniMemcardSection->Get("DefaultMemcardB", &(DefaultMemcard[SLOT_B]), ""); iniMemcardSection->Get("DefaultIOFolder", &DefaultIOPath, "/Users/GC"); iniMemcardSection->Get("Use Pages", &mcmSettings.usePages, true); iniMemcardSection->Get("cBanner", &mcmSettings.column[COLUMN_BANNER], true); iniMemcardSection->Get("cTitle", &mcmSettings.column[COLUMN_TITLE], true); iniMemcardSection->Get("cComment", &mcmSettings.column[COLUMN_COMMENT], true); iniMemcardSection->Get("cIcon", &mcmSettings.column[COLUMN_ICON], true); iniMemcardSection->Get("cBlocks", &mcmSettings.column[COLUMN_BLOCKS], true); iniMemcardSection->Get("cFirst Block", &mcmSettings.column[COLUMN_FIRSTBLOCK], true); mcmSettings.column[NUMBER_OF_COLUMN] = false; for (int i = COLUMN_GAMECODE; i < NUMBER_OF_COLUMN; i++) { mcmSettings.column[i] = mcmSettings.column[NUMBER_OF_COLUMN]; } return true; } return false; } bool CMemcardManager::SaveSettings() { MemcardManagerIni.Load(File::GetUserPath(F_DOLPHINCONFIG_IDX)); iniMemcardSection = MemcardManagerIni.GetOrCreateSection("MemcardManager"); iniMemcardSection->Set("Items per page", itemsPerPage, 16); iniMemcardSection->Set("DefaultMemcardA", DefaultMemcard[SLOT_A], ""); iniMemcardSection->Set("DefaultMemcardB", DefaultMemcard[SLOT_B], ""); iniMemcardSection->Set("Use Pages", mcmSettings.usePages, true); iniMemcardSection->Set("cBanner", mcmSettings.column[COLUMN_BANNER], true); iniMemcardSection->Set("cTitle", mcmSettings.column[COLUMN_TITLE], true); iniMemcardSection->Set("cComment", mcmSettings.column[COLUMN_COMMENT], true); iniMemcardSection->Set("cIcon", mcmSettings.column[COLUMN_ICON], true); iniMemcardSection->Set("cBlocks", mcmSettings.column[COLUMN_BLOCKS], true); iniMemcardSection->Set("cFirst Block", mcmSettings.column[COLUMN_FIRSTBLOCK], true); return MemcardManagerIni.Save(File::GetUserPath(F_DOLPHINCONFIG_IDX)); } void CMemcardManager::CreateGUIControls() { // Create the controls for both memory cards const int space5 = FromDIP(5); const char* ARROW[2] = {"<-", "->"}; m_ConvertToGci = new wxButton(this, ID_CONVERTTOGCI, _("Convert to GCI")); wxStaticBoxSizer* sMemcard[2]; for (int slot = SLOT_A; slot <= SLOT_B; slot++) { m_CopyFrom[slot] = new wxButton(this, ID_COPYFROM_A + slot, wxString::Format(_("%1$sCopy%1$s"), ARROW[slot ? 0 : 1])); m_SaveImport[slot] = new wxButton(this, ID_SAVEIMPORT_A + slot, wxString::Format(_("%sImport GCI%s"), ARROWS)); m_SaveExport[slot] = new wxButton(this, ID_SAVEEXPORT_A + slot, wxString::Format(_("%sExport GCI%s"), ARROWS)); m_Delete[slot] = new wxButton(this, ID_DELETE_A + slot, wxString::Format(_("%sDelete%s"), ARROWS)); m_PrevPage[slot] = new wxButton(this, ID_PREVPAGE_A + slot, _("Prev Page")); m_NextPage[slot] = new wxButton(this, ID_NEXTPAGE_A + slot, _("Next Page")); t_Status[slot] = new wxStaticText(this, 0, wxEmptyString, wxDefaultPosition, wxDefaultSize, 0, wxEmptyString); wxFlexGridSizer* const paging_sizer = new wxFlexGridSizer(3, wxSize(space5, space5)); paging_sizer->AddGrowableCol(1); paging_sizer->Add(m_PrevPage[slot], 0, wxEXPAND); paging_sizer->Add(t_Status[slot], 0, wxALIGN_CENTER); paging_sizer->Add(m_NextPage[slot], 0, wxEXPAND); m_MemcardPath[slot] = new wxFilePickerCtrl( this, ID_MEMCARDPATH_A + slot, StrToWxStr(File::GetUserPath(D_GCUSER_IDX)), _("Choose a memory card:"), _("GameCube Memory Cards (*.raw,*.gcp)") + wxString("|*.raw;*.gcp"), wxDefaultPosition, wxDefaultSize, wxFLP_USE_TEXTCTRL | wxFLP_OPEN); m_MemcardList[slot] = new CMemcardListCtrl( this, ID_MEMCARDLIST_A + slot, wxDefaultPosition, FromDIP(wxSize(350, 400)), wxLC_REPORT | wxSUNKEN_BORDER | wxLC_ALIGN_LEFT | wxLC_SINGLE_SEL, mcmSettings); m_MemcardList[slot]->AssignImageList( new wxImageList(m_image_list_size.GetWidth(), m_image_list_size.GetHeight()), wxIMAGE_LIST_SMALL); sMemcard[slot] = new wxStaticBoxSizer(wxVERTICAL, this, _("Memory Card") + wxString::Format(" %c", 'A' + slot)); sMemcard[slot]->AddSpacer(space5); sMemcard[slot]->Add(m_MemcardPath[slot], 0, wxEXPAND | wxLEFT | wxRIGHT, space5); sMemcard[slot]->AddSpacer(space5); sMemcard[slot]->Add(m_MemcardList[slot], 1, wxEXPAND | wxLEFT | wxRIGHT, space5); sMemcard[slot]->AddSpacer(space5); sMemcard[slot]->Add(paging_sizer, 0, wxEXPAND | wxLEFT | wxRIGHT, space5); sMemcard[slot]->AddSpacer(space5); } wxBoxSizer* const sButtons = new wxBoxSizer(wxVERTICAL); sButtons->AddStretchSpacer(2); sButtons->Add(m_CopyFrom[SLOT_B], 0, wxEXPAND); sButtons->Add(m_CopyFrom[SLOT_A], 0, wxEXPAND); sButtons->AddStretchSpacer(); sButtons->Add(m_SaveImport[SLOT_A], 0, wxEXPAND); sButtons->Add(m_SaveExport[SLOT_A], 0, wxEXPAND); sButtons->AddStretchSpacer(); sButtons->Add(m_ConvertToGci, 0, wxEXPAND); sButtons->AddStretchSpacer(); sButtons->Add(m_SaveImport[SLOT_B], 0, wxEXPAND); sButtons->Add(m_SaveExport[SLOT_B], 0, wxEXPAND); sButtons->AddStretchSpacer(); sButtons->Add(m_Delete[SLOT_A], 0, wxEXPAND); sButtons->Add(m_Delete[SLOT_B], 0, wxEXPAND); sButtons->AddStretchSpacer(); sButtons->Add(new wxButton(this, wxID_OK, _("Close")), 0, wxEXPAND); sButtons->AddStretchSpacer(); wxBoxSizer* const sMain = new wxBoxSizer(wxHORIZONTAL); sMain->AddSpacer(space5); sMain->Add(sMemcard[SLOT_A], 1, wxEXPAND | wxTOP | wxBOTTOM, space5); sMain->AddSpacer(space5); sMain->Add(sButtons, 0, wxEXPAND | wxTOP | wxBOTTOM, space5); sMain->AddSpacer(space5); sMain->Add(sMemcard[SLOT_B], 1, wxEXPAND | wxTOP | wxBOTTOM, space5); sMain->AddSpacer(space5); SetSizerAndFit(sMain); Center(); for (int i = SLOT_A; i <= SLOT_B; 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 (DefaultMemcard[i].length()) { m_MemcardPath[i]->SetPath(StrToWxStr(DefaultMemcard[i])); ChangePath(i); } } } void CMemcardManager::OnPathChange(wxFileDirPickerEvent& event) { ChangePath(event.GetId() - ID_MEMCARDPATH_A); } void CMemcardManager::ChangePath(int slot) { int slot2 = (slot == SLOT_A) ? SLOT_B : SLOT_A; page[slot] = FIRSTPAGE; if (mcmSettings.usePages && m_PrevPage[slot]->IsEnabled()) { m_PrevPage[slot]->Disable(); m_MemcardList[slot]->prevPage = false; } if (!m_MemcardPath[SLOT_A]->GetPath().CmpNoCase(m_MemcardPath[SLOT_B]->GetPath())) { if (m_MemcardPath[slot]->GetPath().length()) wxMessageBox(_("Memory card already opened")); } else { if (m_MemcardPath[slot]->GetPath().length() && ReloadMemcard(WxStrToStr(m_MemcardPath[slot]->GetPath()), slot)) { if (memoryCard[slot2]) { mcmSettings.twoCardsLoaded = true; } m_SaveImport[slot]->Enable(); m_SaveExport[slot]->Enable(); m_Delete[slot]->Enable(); } else { memoryCard[slot].reset(); mcmSettings.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 (mcmSettings.usePages) { m_PrevPage[slot]->Disable(); m_NextPage[slot]->Disable(); } } } m_CopyFrom[SLOT_A]->Enable(mcmSettings.twoCardsLoaded); m_CopyFrom[SLOT_B]->Enable(mcmSettings.twoCardsLoaded); } 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(WxStrToStr(m_MemcardPath[slot]->GetPath()), 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(WxStrToStr(m_MemcardPath[slot]->GetPath()), slot); break; } } void CMemcardManager::OnMenuChange(wxCommandEvent& event) { int _id = event.GetId(); switch (_id) { case ID_MEMCARDPATH_A: case ID_MEMCARDPATH_B: DefaultMemcard[_id - ID_MEMCARDPATH_A] = WxStrToStr(m_MemcardPath[_id - ID_MEMCARDPATH_A]->GetPath()); return; case ID_USEPAGES: mcmSettings.usePages = !mcmSettings.usePages; if (!mcmSettings.usePages) { m_PrevPage[SLOT_A]->Disable(); m_PrevPage[SLOT_B]->Disable(); m_NextPage[SLOT_A]->Disable(); m_NextPage[SLOT_B]->Disable(); m_MemcardList[SLOT_A]->prevPage = m_MemcardList[SLOT_B]->prevPage = false; page[SLOT_A] = page[SLOT_B] = FIRSTPAGE; } break; case NUMBER_OF_COLUMN: for (int i = COLUMN_GAMECODE; i <= NUMBER_OF_COLUMN; i++) { mcmSettings.column[i] = !mcmSettings.column[i]; } break; default: mcmSettings.column[_id] = !mcmSettings.column[_id]; break; } if (memoryCard[SLOT_A]) ReloadMemcard(WxStrToStr(m_MemcardPath[SLOT_A]->GetPath()), SLOT_A); if (memoryCard[SLOT_B]) ReloadMemcard(WxStrToStr(m_MemcardPath[SLOT_B]->GetPath()), SLOT_B); } bool CMemcardManager::CopyDeleteSwitch(u32 error, int slot) { switch (error) { case GCS: wxMessageBox(_("File converted to .gci")); break; case SUCCESS: if (slot != -1) { memoryCard[slot]->FixChecksums(); if (!memoryCard[slot]->Save()) PanicAlertT("File write failed"); page[slot] = FIRSTPAGE; ReloadMemcard(WxStrToStr(m_MemcardPath[slot]->GetPath()), slot); } break; case NOMEMCARD: WxUtils::ShowErrorDialog(_("File is not recognized as a memory card")); break; case OPENFAIL: WxUtils::ShowErrorDialog(_("File could not be opened\nor does not have a valid extension")); break; case OUTOFBLOCKS: if (slot == -1) { WxUtils::ShowErrorDialog(_("Unknown memory card error")); break; } wxMessageBox( wxString::Format(_("Only %d blocks available"), memoryCard[slot]->GetFreeBlocks())); break; case OUTOFDIRENTRIES: WxUtils::ShowErrorDialog(_("No free directory index entries.")); break; case LENGTHFAIL: WxUtils::ShowErrorDialog(_("Imported file has invalid length.")); break; case INVALIDFILESIZE: WxUtils::ShowErrorDialog(_("The save you are trying to copy has an invalid file size.")); break; case TITLEPRESENT: WxUtils::ShowErrorDialog(_("Memory card already has a save for this title.")); break; case SAVFAIL: WxUtils::ShowErrorDialog( _("Imported file has sav extension\nbut does not have a correct header.")); break; case GCSFAIL: WxUtils::ShowErrorDialog( _("Imported file has gsc extension\nbut does not have a correct header.")); break; case FAIL: if (slot == -1) { WxUtils::ShowErrorDialog(_("Export failed")); return false; } WxUtils::ShowErrorDialog(_("Invalid bat.map or dir entry.")); break; case WRITEFAIL: WxUtils::ShowErrorDialog(_("File write failed")); break; case DELETE_FAIL: WxUtils::ShowErrorDialog( _("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 saves to a new memory card\n")); break; default: WxUtils::ShowErrorDialog(_("Unknown memory card error")); break; } SetFocus(); 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; 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]; int index = index_B; switch (event.GetId()) { case ID_COPYFROM_B: slot = SLOT_A; slot2 = SLOT_B; case ID_COPYFROM_A: index = slot2 ? index_B : index_A; index = memoryCard[slot2]->GetFileIndex(index); 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()) { wxMessageBox(_("The checksum was successfully fixed.")); } else { WxUtils::ShowErrorDialog(_("File write failed")); } break; case ID_CONVERTTOGCI: fileName2 = "convert"; case ID_SAVEIMPORT_A: slot = SLOT_A; case ID_SAVEIMPORT_B: { wxString fileName = wxFileSelector( _("Select a save file to import"), (strcmp(DefaultIOPath.c_str(), "/Users/GC") == 0) ? StrToWxStr("") : StrToWxStr(DefaultIOPath), wxEmptyString, wxEmptyString, _("GameCube Savegame files(*.gci;*.gcs;*.sav)") + wxString("|*.gci;*.gcs;*.sav|") + _("Native GCI files(*.gci)") + wxString("|*.gci|") + _("MadCatz Gameshark files(*.gcs)") + wxString("|*.gcs|") + _("Datel MaxDrive/Pro files(*.sav)") + wxString("|*.sav"), wxFD_OPEN | wxFD_FILE_MUST_EXIST, this); if (!fileName.empty() && !fileName2.empty()) { wxString temp2 = wxFileSelector(_("Save GCI as..."), wxEmptyString, wxEmptyString, ".gci", _("GCI File(*.gci)") + wxString("|*.gci"), wxFD_OVERWRITE_PROMPT | wxFD_SAVE, this); if (temp2.empty()) break; fileName2 = WxStrToStr(temp2); } if (fileName.length() > 0) { CopyDeleteSwitch(memoryCard[slot]->ImportGci(WxStrToStr(fileName), fileName2), slot); } } break; case ID_SAVEEXPORT_A: slot = SLOT_A; index = index_A; case ID_SAVEEXPORT_B: index = memoryCard[slot]->GetFileIndex(index); if (index != wxNOT_FOUND) { std::string gciFilename; if (!memoryCard[slot]->GCI_FileName(index, gciFilename)) { wxMessageBox(_("Invalid index"), _("Error")); return; } wxString fileName = wxFileSelector( _("Export save as..."), StrToWxStr(DefaultIOPath), StrToWxStr(gciFilename), ".gci", _("Native GCI files(*.gci)") + wxString("|*.gci|") + _("MadCatz Gameshark files(*.gcs)") + wxString("|*.gcs|") + _("Datel MaxDrive/Pro files(*.sav)") + wxString("|*.sav"), wxFD_OVERWRITE_PROMPT | wxFD_SAVE, this); if (fileName.length() > 0) { if (!CopyDeleteSwitch(memoryCard[slot]->ExportGci(index, WxStrToStr(fileName), ""), -1)) { File::Delete(WxStrToStr(fileName)); } } } break; case ID_EXPORTALL_A: slot = SLOT_A; case ID_EXPORTALL_B: { std::string path1, path2, mpath; mpath = WxStrToStr(m_MemcardPath[slot]->GetPath()); SplitPath(mpath, &path1, &path2, nullptr); path1 += path2; File::CreateDir(path1); int answer = wxMessageBox( wxString::Format( _("Warning: This will overwrite any existing saves that are in the folder:\n" "%s\nand have the same name as a file on your memory card\nContinue?"), path1.c_str()), _("Warning"), wxYES_NO); if (answer == wxYES) { 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: index = memoryCard[slot]->GetFileIndex(index); if (index != wxNOT_FOUND) { CopyDeleteSwitch(memoryCard[slot]->RemoveFile(index), slot); } break; } } bool CMemcardManager::ReloadMemcard(const std::string& fileName, int card) { // TODO: add error checking and animate icons memoryCard[card] = std::make_unique(fileName); if (!memoryCard[card]->IsValid()) return false; int j; wxString wxTitle, wxComment, wxBlock, wxFirstBlock, wxLabel; m_MemcardList[card]->Hide(); m_MemcardList[card]->ClearAll(); m_MemcardList[card]->InsertColumn(COLUMN_BANNER, _("Banner")); m_MemcardList[card]->InsertColumn(COLUMN_TITLE, _("Title")); m_MemcardList[card]->InsertColumn(COLUMN_COMMENT, _("Comment")); m_MemcardList[card]->InsertColumn(COLUMN_ICON, _("Icon")); m_MemcardList[card]->InsertColumn(COLUMN_BLOCKS, _("Blocks")); m_MemcardList[card]->InsertColumn(COLUMN_FIRSTBLOCK, _("First Block")); wxImageList* list = m_MemcardList[card]->GetImageList(wxIMAGE_LIST_SMALL); list->RemoveAll(); u8 nFiles = memoryCard[card]->GetNumFiles(); std::vector images(nFiles * 2); static constexpr unsigned int BANNER_WIDTH = 96; static constexpr unsigned int ANIM_FRAME_WIDTH = 32; static constexpr unsigned int IMAGE_HEIGHT = 32; static constexpr unsigned int ANIM_MAX_FRAMES = 8; std::vector pxdata(BANNER_WIDTH * IMAGE_HEIGHT); std::vector anim_delay(ANIM_MAX_FRAMES); std::vector anim_data(ANIM_FRAME_WIDTH * IMAGE_HEIGHT * ANIM_MAX_FRAMES); for (u8 i = 0; i < nFiles; i++) { u8 file_index = memoryCard[card]->GetFileIndex(i); u32 num_frames = memoryCard[card]->ReadAnimRGBA8(file_index, anim_data.data(), anim_delay.data()); // Decode Save File Animation wxImage anim_img_strip; if (num_frames > 0) { unsigned int frames = BANNER_WIDTH / ANIM_FRAME_WIDTH; if (num_frames < frames) { frames = num_frames; // Clear unused frame's pixels from the buffer. std::fill(pxdata.begin(), pxdata.end(), 0); } for (unsigned int f = 0; f < frames; ++f) { for (unsigned int y = 0; y < IMAGE_HEIGHT; ++y) { for (unsigned int x = 0; x < ANIM_FRAME_WIDTH; ++x) { // NOTE: pxdata is stacked horizontal, anim_data is stacked vertical pxdata[y * BANNER_WIDTH + f * ANIM_FRAME_WIDTH + x] = anim_data[f * ANIM_FRAME_WIDTH * IMAGE_HEIGHT + y * IMAGE_HEIGHT + x]; } } } anim_img_strip = wxImageFromMemoryRGBA(pxdata.data(), BANNER_WIDTH, IMAGE_HEIGHT); } else { // No Animation found, use an empty placeholder instead. anim_img_strip.Create(BANNER_WIDTH, IMAGE_HEIGHT, false); anim_img_strip.Clear(0xFF); anim_img_strip.SetMaskColour(0xFF, 0xFF, 0xFF); } // Decode Banner if it exists { wxImage image; if (memoryCard[card]->ReadBannerRGBA8(file_index, pxdata.data())) { image = wxImageFromMemoryRGBA(pxdata.data(), BANNER_WIDTH, IMAGE_HEIGHT); } else { // Use first frame of animation instead. image = anim_img_strip.Size(wxSize(ANIM_FRAME_WIDTH, IMAGE_HEIGHT), wxPoint(0, 0)); } images[i * 2] = list->Add(WxUtils::ScaleImageToBitmap(image, this, m_image_list_size)); } images[i * 2 + 1] = list->Add(WxUtils::ScaleImageToBitmap(anim_img_strip, this, m_image_list_size)); } int pagesMax = (mcmSettings.usePages) ? (page[card] + 1) * itemsPerPage : 128; for (j = page[card] * itemsPerPage; (j < nFiles) && (j < pagesMax); j++) { u16 blocks; u16 firstblock; u8 fileIndex = memoryCard[card]->GetFileIndex(j); int index = m_MemcardList[card]->InsertItem(j, wxEmptyString); m_MemcardList[card]->SetItem(index, COLUMN_BANNER, wxEmptyString); std::string title = memoryCard[card]->GetSaveComment1(fileIndex); std::string comment = memoryCard[card]->GetSaveComment2(fileIndex); auto const string_decoder = memoryCard[card]->IsShiftJIS() ? SHIFTJISToUTF8 : CP1252ToUTF8; wxTitle = StrToWxStr(string_decoder(title)); wxComment = StrToWxStr(string_decoder(comment)); m_MemcardList[card]->SetItem(index, COLUMN_TITLE, wxTitle); m_MemcardList[card]->SetItem(index, COLUMN_COMMENT, wxComment); blocks = memoryCard[card]->DEntry_BlockCount(fileIndex); if (blocks == 0xFFFF) blocks = 0; wxBlock.Printf("%10d", blocks); m_MemcardList[card]->SetItem(index, COLUMN_BLOCKS, wxBlock); firstblock = memoryCard[card]->DEntry_FirstBlock(fileIndex); // if (firstblock == 0xFFFF) firstblock = 3; // to make firstblock -1 wxFirstBlock.Printf("%15d", firstblock); 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 (mcmSettings.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; } } // Automatic column width and then show the list for (int i = COLUMN_BANNER; i <= COLUMN_FIRSTBLOCK; i++) { if (mcmSettings.column[i]) m_MemcardList[card]->SetColumnWidth(i, wxLIST_AUTOSIZE); else m_MemcardList[card]->SetColumnWidth(i, 0); } m_MemcardList[card]->Show(); wxLabel.Printf(_("%u Free Blocks; %u Free Dir Entries"), memoryCard[card]->GetFreeBlocks(), DIRLEN - nFiles); t_Status[card]->SetLabel(wxLabel); // Done so text doesn't overlap the UI. this->Fit(); 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 = GetId() - ID_MEMCARDLIST_A; popupMenu.Append(ID_COPYFROM_A + slot, wxString::Format(_("Copy to Memory Card %c"), 'B' - slot)); popupMenu.Append(ID_DELETE_A + slot, _("Delete Save")); popupMenu.Append(ID_SAVEIMPORT_A + slot, _("Import Save")); popupMenu.Append(ID_SAVEEXPORT_A + slot, _("Export Save")); popupMenu.Append(ID_EXPORTALL_A + slot, _("Export all saves")); popupMenu.FindItem(ID_COPYFROM_A + slot)->Enable(mgr_settings.twoCardsLoaded); popupMenu.AppendSeparator(); popupMenu.Append(ID_FIXCHECKSUM_A + slot, _("Fix Checksums")); popupMenu.Append(ID_PREVPAGE_A + slot, _("Previous Page")); popupMenu.Append(ID_NEXTPAGE_A + slot, _("Next Page")); popupMenu.Append(ID_MEMCARDPATH_A + slot, wxString::Format(_("Set as default Memory Card %c"), 'A' + slot)); popupMenu.AppendCheckItem(ID_USEPAGES, _("Enable pages")); popupMenu.FindItem(ID_PREVPAGE_A + slot)->Enable(prevPage && mgr_settings.usePages); popupMenu.FindItem(ID_NEXTPAGE_A + slot)->Enable(nextPage && mgr_settings.usePages); popupMenu.FindItem(ID_USEPAGES)->Check(mgr_settings.usePages); popupMenu.AppendSeparator(); // popupMenu->AppendCheckItem(COLUMN_BANNER, _("Show save banner")); popupMenu.AppendCheckItem(COLUMN_TITLE, _("Show save title")); popupMenu.AppendCheckItem(COLUMN_COMMENT, _("Show save comment")); popupMenu.AppendCheckItem(COLUMN_ICON, _("Show save icon")); popupMenu.AppendCheckItem(COLUMN_BLOCKS, _("Show save blocks")); popupMenu.AppendCheckItem(COLUMN_FIRSTBLOCK, _("Show first block")); // for (int i = COLUMN_BANNER; i <= COLUMN_FIRSTBLOCK; i++) for (int i = COLUMN_TITLE; i <= COLUMN_FIRSTBLOCK; i++) { popupMenu.FindItem(i)->Check(mgr_settings.column[i]); } } PopupMenu(&popupMenu); }