2009-07-28 21:32:10 +00:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
// 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 <string>
|
|
|
|
#include <vector>
|
|
|
|
#include <wx/mstream.h>
|
|
|
|
|
|
|
|
#include "Globals.h"
|
|
|
|
#include "FileUtil.h"
|
|
|
|
#include "ISOFile.h"
|
|
|
|
#include "StringUtil.h"
|
|
|
|
|
|
|
|
#include "Filesystem.h"
|
|
|
|
#include "BannerLoader.h"
|
|
|
|
#include "FileSearch.h"
|
|
|
|
#include "CompressedBlob.h"
|
|
|
|
#include "ChunkFile.h"
|
|
|
|
#include "../resources/no_banner.cpp"
|
|
|
|
|
2009-06-06 07:36:22 +00:00
|
|
|
#define CACHE_REVISION 0x109
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
#define DVD_BANNER_WIDTH 96
|
|
|
|
#define DVD_BANNER_HEIGHT 32
|
|
|
|
|
|
|
|
static u32 g_ImageTemp[DVD_BANNER_WIDTH * DVD_BANNER_HEIGHT];
|
|
|
|
|
2009-02-03 15:03:34 +00:00
|
|
|
GameListItem::GameListItem(const std::string& _rFileName)
|
2008-12-08 05:30:24 +00:00
|
|
|
: m_FileName(_rFileName)
|
|
|
|
, m_FileSize(0)
|
|
|
|
, m_Valid(false)
|
|
|
|
, m_BlobCompressed(false)
|
|
|
|
, m_pImage(NULL)
|
|
|
|
, m_ImageSize(0)
|
|
|
|
{
|
2009-02-03 15:03:34 +00:00
|
|
|
if (LoadFromCache())
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
|
|
|
m_Valid = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DiscIO::IVolume* pVolume = DiscIO::CreateVolumeFromFilename(_rFileName);
|
|
|
|
|
|
|
|
if (pVolume != NULL)
|
|
|
|
{
|
2009-06-07 02:54:07 +00:00
|
|
|
if (!DiscIO::IsVolumeWadFile(pVolume))
|
|
|
|
m_Platform = DiscIO::IsVolumeWiiDisc(pVolume) ? WII_DISC : GAMECUBE_DISC;
|
|
|
|
else
|
|
|
|
m_Platform = WII_WAD;
|
2009-05-27 06:41:01 +00:00
|
|
|
|
2009-02-07 17:31:35 +00:00
|
|
|
m_Company = "N/A";
|
2009-03-07 04:46:06 +00:00
|
|
|
for (int i = 0; i < 6; i++)
|
|
|
|
{
|
|
|
|
m_Name[i] = pVolume->GetName();
|
2010-01-09 22:03:20 +00:00
|
|
|
if(m_Name[i] == "") // Couldn't find the name in the WAD...
|
|
|
|
{
|
|
|
|
std::string FileName;
|
|
|
|
SplitPath(_rFileName, NULL, &FileName, NULL);
|
|
|
|
m_Name[i] = FileName; // Then just display the filename... Better than something like "No Name"
|
|
|
|
}
|
2009-03-07 04:46:06 +00:00
|
|
|
m_Description[i] = "No Description";
|
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
m_Country = pVolume->GetCountry();
|
|
|
|
m_FileSize = File::GetSize(_rFileName.c_str());
|
|
|
|
m_VolumeSize = pVolume->GetSize();
|
2009-06-06 07:36:22 +00:00
|
|
|
|
2008-12-08 05:30:24 +00:00
|
|
|
m_UniqueID = pVolume->GetUniqueID();
|
|
|
|
m_BlobCompressed = DiscIO::IsCompressedBlob(_rFileName.c_str());
|
|
|
|
|
|
|
|
// check if we can get some infos from the banner file too
|
|
|
|
DiscIO::IFileSystem* pFileSystem = DiscIO::CreateFileSystem(pVolume);
|
|
|
|
|
2009-06-07 02:54:07 +00:00
|
|
|
if (pFileSystem != NULL || m_Platform == WII_WAD)
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2009-06-07 02:54:07 +00:00
|
|
|
DiscIO::IBannerLoader* pBannerLoader = DiscIO::CreateBannerLoader(*pFileSystem, pVolume);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
if (pBannerLoader != NULL)
|
|
|
|
{
|
|
|
|
if (pBannerLoader->IsValid())
|
|
|
|
{
|
2009-02-03 15:03:34 +00:00
|
|
|
pBannerLoader->GetName(m_Name); //m_Country == DiscIO::IVolume::COUNTRY_JAP ? 1 : 0);
|
2008-12-08 05:30:24 +00:00
|
|
|
pBannerLoader->GetCompany(m_Company);
|
2009-02-03 15:03:34 +00:00
|
|
|
pBannerLoader->GetDescription(m_Description);
|
2008-12-08 05:30:24 +00:00
|
|
|
if (pBannerLoader->GetBanner(g_ImageTemp))
|
|
|
|
{
|
|
|
|
m_ImageSize = DVD_BANNER_WIDTH * DVD_BANNER_HEIGHT * 3;
|
2010-02-12 19:28:51 +00:00
|
|
|
//use malloc(), since wxImage::Create below calls free() afterwards.
|
|
|
|
m_pImage = (u8*)malloc(m_ImageSize);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
for (size_t i = 0; i < DVD_BANNER_WIDTH * DVD_BANNER_HEIGHT; i++)
|
|
|
|
{
|
|
|
|
m_pImage[i * 3 + 0] = (g_ImageTemp[i] & 0xFF0000) >> 16;
|
|
|
|
m_pImage[i * 3 + 1] = (g_ImageTemp[i] & 0x00FF00) >> 8;
|
|
|
|
m_pImage[i * 3 + 2] = (g_ImageTemp[i] & 0x0000FF) >> 0;
|
2009-06-06 07:36:22 +00:00
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
delete pBannerLoader;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete pFileSystem;
|
|
|
|
}
|
|
|
|
|
|
|
|
delete pVolume;
|
|
|
|
|
|
|
|
m_Valid = true;
|
|
|
|
|
2010-06-03 20:37:32 +00:00
|
|
|
// If not Gamecube, create a cache file only if we have an image.
|
2008-12-08 05:30:24 +00:00
|
|
|
// Wii isos create their images after you have generated the first savegame
|
2010-06-03 20:37:32 +00:00
|
|
|
if (m_Platform == GAMECUBE_DISC || m_pImage)
|
2008-12-08 05:30:24 +00:00
|
|
|
SaveToCache();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (m_pImage)
|
|
|
|
{
|
|
|
|
m_Image.Create(DVD_BANNER_WIDTH, DVD_BANNER_HEIGHT, m_pImage);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// default banner
|
|
|
|
wxMemoryInputStream istream(no_banner_png, sizeof no_banner_png);
|
|
|
|
wxImage iNoBanner(istream, wxBITMAP_TYPE_PNG);
|
|
|
|
m_Image = iNoBanner;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GameListItem::~GameListItem()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GameListItem::LoadFromCache()
|
|
|
|
{
|
|
|
|
return CChunkFileReader::Load<GameListItem>(CreateCacheFilename(), CACHE_REVISION, *this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListItem::SaveToCache()
|
|
|
|
{
|
2010-02-02 21:56:29 +00:00
|
|
|
if (!File::IsDirectory(File::GetUserPath(D_CACHE_IDX)))
|
2008-12-08 05:30:24 +00:00
|
|
|
{
|
2010-02-02 21:56:29 +00:00
|
|
|
File::CreateDir(File::GetUserPath(D_CACHE_IDX));
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CChunkFileReader::Save<GameListItem>(CreateCacheFilename(), CACHE_REVISION, *this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GameListItem::DoState(PointerWrap &p)
|
|
|
|
{
|
2009-02-03 15:03:34 +00:00
|
|
|
p.Do(m_Name[0]); p.Do(m_Name[1]); p.Do(m_Name[2]);
|
|
|
|
p.Do(m_Name[3]); p.Do(m_Name[4]); p.Do(m_Name[5]);
|
2008-12-08 05:30:24 +00:00
|
|
|
p.Do(m_Company);
|
2009-02-03 15:03:34 +00:00
|
|
|
p.Do(m_Description[0]); p.Do(m_Description[1]); p.Do(m_Description[2]);
|
|
|
|
p.Do(m_Description[3]); p.Do(m_Description[4]); p.Do(m_Description[5]);
|
2008-12-08 05:30:24 +00:00
|
|
|
p.Do(m_UniqueID);
|
|
|
|
p.Do(m_FileSize);
|
|
|
|
p.Do(m_VolumeSize);
|
|
|
|
p.Do(m_Country);
|
|
|
|
p.Do(m_BlobCompressed);
|
|
|
|
p.DoBuffer(&m_pImage, m_ImageSize);
|
2009-06-06 07:36:22 +00:00
|
|
|
p.Do(m_Platform);
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string GameListItem::CreateCacheFilename()
|
|
|
|
{
|
|
|
|
std::string Filename;
|
|
|
|
SplitPath(m_FileName, NULL, &Filename, NULL);
|
2009-04-08 16:38:41 +00:00
|
|
|
|
2009-04-28 02:30:50 +00:00
|
|
|
if (Filename.empty()) return Filename; // Disc Drive
|
2009-04-08 16:38:41 +00:00
|
|
|
// We add gcz to the cache file if the file is compressed to avoid it reading
|
|
|
|
// the uncompressed file's cache if it has the same name, but not the same ext.
|
|
|
|
if (DiscIO::IsCompressedBlob(m_FileName.c_str()))
|
|
|
|
Filename.append(".gcz");
|
2008-12-08 05:30:24 +00:00
|
|
|
Filename.append(".cache");
|
|
|
|
|
2010-02-02 21:56:29 +00:00
|
|
|
std::string fullname(std::string(File::GetUserPath(D_CACHE_IDX)));
|
2008-12-08 05:30:24 +00:00
|
|
|
fullname += Filename;
|
|
|
|
return fullname;
|
|
|
|
}
|
2009-02-03 15:03:34 +00:00
|
|
|
|
|
|
|
const std::string& GameListItem::GetDescription(int index) const
|
|
|
|
{
|
|
|
|
if ((index >=0) && (index < 6))
|
|
|
|
{
|
|
|
|
return m_Description[index];
|
|
|
|
}
|
|
|
|
return m_Description[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& GameListItem::GetName(int index) const
|
|
|
|
{
|
|
|
|
if ((index >=0) && (index < 6))
|
|
|
|
{
|
|
|
|
return m_Name[index];
|
|
|
|
}
|
|
|
|
return m_Name[0];
|
|
|
|
}
|
2009-04-28 02:30:50 +00:00
|
|
|
|
2009-05-02 18:06:42 +00:00
|
|
|
const std::string GameListItem::GetWiiFSPath() const
|
|
|
|
{
|
|
|
|
DiscIO::IVolume *Iso = DiscIO::CreateVolumeFromFilename(m_FileName);
|
|
|
|
|
2010-01-14 07:19:10 +00:00
|
|
|
std::string ret("NULL");
|
2009-05-02 18:06:42 +00:00
|
|
|
if (Iso != NULL)
|
|
|
|
{
|
2009-06-07 02:54:07 +00:00
|
|
|
if (DiscIO::IsVolumeWiiDisc(Iso) || DiscIO::IsVolumeWadFile(Iso))
|
2009-05-02 18:06:42 +00:00
|
|
|
{
|
|
|
|
char Path[250];
|
|
|
|
u64 Title;
|
|
|
|
|
2010-01-14 07:19:10 +00:00
|
|
|
Iso->GetTitleID((u8*)&Title);
|
2009-05-02 18:06:42 +00:00
|
|
|
Title = Common::swap64(Title);
|
|
|
|
|
2010-02-02 21:56:29 +00:00
|
|
|
sprintf(Path, "%stitle/%08x/%08x/data/", File::GetUserPath(D_WIIUSER_IDX), (u32)(Title>>32), (u32)Title);
|
2009-05-02 18:06:42 +00:00
|
|
|
|
|
|
|
if (!File::Exists(Path))
|
|
|
|
File::CreateFullPath(Path);
|
|
|
|
|
2010-02-03 15:13:13 +00:00
|
|
|
if (Path[0] == '.')
|
|
|
|
ret = std::string(wxGetCwd().mb_str()) + std::string(Path).substr(strlen(ROOT_DIR));
|
|
|
|
else
|
|
|
|
ret = std::string(Path);
|
2009-05-02 18:06:42 +00:00
|
|
|
}
|
2010-01-14 07:19:10 +00:00
|
|
|
delete Iso;
|
2009-05-02 18:06:42 +00:00
|
|
|
}
|
|
|
|
|
2010-01-14 07:19:10 +00:00
|
|
|
return ret;
|
2009-05-02 18:06:42 +00:00
|
|
|
}
|
|
|
|
|