2021-12-09 10:08:43 +00:00
|
|
|
/* PCSX2 - PS2 Emulator for PCs
|
2023-05-13 04:30:41 +00:00
|
|
|
* Copyright (C) 2002-2023 PCSX2 Dev Team
|
2021-12-09 10:08:43 +00:00
|
|
|
*
|
|
|
|
* PCSX2 is free software: you can redistribute it and/or modify it under the terms
|
|
|
|
* of the GNU Lesser General Public License as published by the Free Software Found-
|
|
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* PCSX2 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 PCSX2.
|
|
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
2023-05-13 04:30:41 +00:00
|
|
|
|
2021-12-09 10:08:43 +00:00
|
|
|
#include "GameDatabase.h"
|
2023-05-13 04:30:41 +00:00
|
|
|
|
2021-12-09 10:08:43 +00:00
|
|
|
#include "common/Pcsx2Defs.h"
|
2023-05-13 04:30:41 +00:00
|
|
|
|
2021-12-09 10:08:43 +00:00
|
|
|
#include <ctime>
|
2022-09-03 11:29:02 +00:00
|
|
|
#include <functional>
|
2021-12-09 10:08:43 +00:00
|
|
|
#include <memory>
|
|
|
|
#include <mutex>
|
|
|
|
#include <optional>
|
|
|
|
#include <string>
|
|
|
|
#include <string_view>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
class ProgressCallback;
|
|
|
|
|
|
|
|
struct VMBootParameters;
|
|
|
|
|
|
|
|
namespace GameList
|
|
|
|
{
|
|
|
|
enum class EntryType
|
|
|
|
{
|
|
|
|
PS2Disc,
|
|
|
|
PS1Disc,
|
|
|
|
ELF,
|
2023-07-09 08:15:55 +00:00
|
|
|
Invalid,
|
2021-12-09 10:08:43 +00:00
|
|
|
Count
|
|
|
|
};
|
|
|
|
|
|
|
|
enum class Region
|
|
|
|
{
|
2022-05-30 23:45:05 +00:00
|
|
|
NTSC_B,
|
|
|
|
NTSC_C,
|
|
|
|
NTSC_HK,
|
2021-12-09 10:08:43 +00:00
|
|
|
NTSC_J,
|
2022-05-30 23:45:05 +00:00
|
|
|
NTSC_K,
|
|
|
|
NTSC_T,
|
|
|
|
NTSC_U,
|
2021-12-09 10:08:43 +00:00
|
|
|
Other,
|
2022-06-01 14:49:09 +00:00
|
|
|
PAL_A,
|
2022-06-02 02:31:38 +00:00
|
|
|
PAL_AU,
|
|
|
|
PAL_AF,
|
2022-05-30 23:45:05 +00:00
|
|
|
PAL_BE,
|
|
|
|
PAL_E,
|
2022-06-01 14:49:09 +00:00
|
|
|
PAL_F,
|
2022-06-02 02:31:38 +00:00
|
|
|
PAL_FI,
|
2022-06-01 14:49:09 +00:00
|
|
|
PAL_G,
|
2022-06-02 02:31:38 +00:00
|
|
|
PAL_GR,
|
2022-06-01 14:49:09 +00:00
|
|
|
PAL_I,
|
2022-06-02 02:31:38 +00:00
|
|
|
PAL_IN,
|
2022-05-30 23:45:05 +00:00
|
|
|
PAL_M,
|
|
|
|
PAL_NL,
|
|
|
|
PAL_NO,
|
|
|
|
PAL_P,
|
2023-07-02 09:40:13 +00:00
|
|
|
PAL_PL,
|
2022-05-30 23:45:05 +00:00
|
|
|
PAL_R,
|
2022-06-02 02:31:38 +00:00
|
|
|
PAL_S,
|
2022-05-30 23:45:05 +00:00
|
|
|
PAL_SC,
|
2022-06-01 14:49:09 +00:00
|
|
|
PAL_SW,
|
2022-06-02 02:31:38 +00:00
|
|
|
PAL_SWI,
|
2022-05-30 23:45:05 +00:00
|
|
|
PAL_UK,
|
2021-12-09 10:08:43 +00:00
|
|
|
Count
|
|
|
|
};
|
|
|
|
|
|
|
|
using CompatibilityRating = GameDatabaseSchema::Compatibility;
|
|
|
|
static constexpr u32 CompatibilityRatingCount = static_cast<u32>(GameDatabaseSchema::Compatibility::Perfect) + 1u;
|
|
|
|
|
|
|
|
struct Entry
|
|
|
|
{
|
|
|
|
EntryType type = EntryType::PS2Disc;
|
|
|
|
Region region = Region::Other;
|
|
|
|
|
|
|
|
std::string path;
|
|
|
|
std::string serial;
|
|
|
|
std::string title;
|
2023-07-04 21:59:43 +00:00
|
|
|
std::string title_sort;
|
|
|
|
std::string title_en;
|
2021-12-09 10:08:43 +00:00
|
|
|
u64 total_size = 0;
|
|
|
|
std::time_t last_modified_time = 0;
|
2022-10-22 04:52:01 +00:00
|
|
|
std::time_t last_played_time = 0;
|
|
|
|
std::time_t total_played_time = 0;
|
2021-12-09 10:08:43 +00:00
|
|
|
|
2023-09-09 22:36:10 +00:00
|
|
|
const std::string& GetTitle(bool force_en = false) const
|
|
|
|
{
|
|
|
|
return title_en.empty() || !force_en ? title : title_en;
|
|
|
|
}
|
|
|
|
const std::string& GetTitleSort(bool force_en = false) const
|
|
|
|
{
|
|
|
|
// If there's a separate EN title, then title_sort is in the wrong language and we can't use it
|
|
|
|
if (force_en && !title_en.empty())
|
|
|
|
return title_en;
|
|
|
|
return title_sort.empty() ? title : title_sort;
|
|
|
|
}
|
2023-07-04 21:59:43 +00:00
|
|
|
|
2021-12-09 10:08:43 +00:00
|
|
|
u32 crc = 0;
|
|
|
|
|
|
|
|
CompatibilityRating compatibility_rating = CompatibilityRating::Unknown;
|
2022-06-28 13:34:45 +00:00
|
|
|
|
|
|
|
__fi bool IsDisc() const { return (type == EntryType::PS1Disc || type == EntryType::PS2Disc); }
|
2021-12-09 10:08:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const char* EntryTypeToString(EntryType type);
|
2022-07-09 08:52:33 +00:00
|
|
|
const char* EntryTypeToDisplayString(EntryType type);
|
2022-05-15 08:12:28 +00:00
|
|
|
const char* RegionToString(Region region);
|
2021-12-09 10:08:43 +00:00
|
|
|
const char* EntryCompatibilityRatingToString(CompatibilityRating rating);
|
|
|
|
|
|
|
|
/// Fills in boot parameters (iso or elf) based on the game list entry.
|
|
|
|
void FillBootParametersForEntry(VMBootParameters* params, const Entry* entry);
|
|
|
|
|
2022-05-03 04:26:49 +00:00
|
|
|
/// Populates a game list entry struct with information from the iso/elf.
|
|
|
|
/// Do *not* call while the system is running, it will mess with CDVD state.
|
|
|
|
bool PopulateEntryFromPath(const std::string& path, GameList::Entry* entry);
|
|
|
|
|
2021-12-09 10:08:43 +00:00
|
|
|
// Game list access. It's the caller's responsibility to hold the lock while manipulating the entry in any way.
|
|
|
|
std::unique_lock<std::recursive_mutex> GetLock();
|
|
|
|
const Entry* GetEntryByIndex(u32 index);
|
|
|
|
const Entry* GetEntryForPath(const char* path);
|
2022-03-12 13:57:10 +00:00
|
|
|
const Entry* GetEntryByCRC(u32 crc);
|
2021-12-09 10:08:43 +00:00
|
|
|
const Entry* GetEntryBySerialAndCRC(const std::string_view& serial, u32 crc);
|
|
|
|
u32 GetEntryCount();
|
|
|
|
|
2022-06-23 11:58:14 +00:00
|
|
|
/// Populates the game list with files in the configured directories.
|
|
|
|
/// If invalidate_cache is set, all files will be re-scanned.
|
|
|
|
/// If only_cache is set, no new files will be scanned, only those present in the cache.
|
|
|
|
void Refresh(bool invalidate_cache, bool only_cache = false, ProgressCallback* progress = nullptr);
|
2021-12-09 10:08:43 +00:00
|
|
|
|
2022-12-04 08:03:30 +00:00
|
|
|
/// Re-scans a single entry in the game list.
|
|
|
|
bool RescanPath(const std::string& path);
|
|
|
|
|
2023-05-13 10:22:46 +00:00
|
|
|
/// Looks up the serial and CRC for a game in the most efficient manner possible (i.e. cache or scan).
|
|
|
|
bool GetSerialAndCRCForFilename(const char* filename, std::string* serial, u32* crc);
|
|
|
|
|
2022-10-22 04:52:01 +00:00
|
|
|
/// Add played time for the specified serial.
|
|
|
|
void AddPlayedTimeForSerial(const std::string& serial, std::time_t last_time, std::time_t add_time);
|
2022-11-22 15:03:36 +00:00
|
|
|
void ClearPlayedTimeForSerial(const std::string& serial);
|
2022-10-22 04:52:01 +00:00
|
|
|
|
|
|
|
/// Returns the total time played for a game. Requires the game to be scanned in the list.
|
|
|
|
std::time_t GetCachedPlayedTimeForSerial(const std::string& serial);
|
|
|
|
|
|
|
|
/// Formats a timestamp to something human readable (e.g. Today, Yesterday, 10/11/12).
|
|
|
|
std::string FormatTimestamp(std::time_t timestamp);
|
|
|
|
|
|
|
|
/// Formats a timespan to something human readable (e.g. 1h2m3s or 1 hour).
|
|
|
|
std::string FormatTimespan(std::time_t timespan, bool long_format = false);
|
|
|
|
|
2021-12-09 10:08:43 +00:00
|
|
|
std::string GetCoverImagePathForEntry(const Entry* entry);
|
2022-09-03 11:29:02 +00:00
|
|
|
std::string GetNewCoverImagePathForEntry(const Entry* entry, const char* new_filename, bool use_serial = false);
|
|
|
|
|
|
|
|
/// Downloads covers using the specified URL templates. By default, covers are saved by title, but this can be changed with
|
|
|
|
/// the use_serial parameter. save_callback optionall takes the entry and the path the new cover is saved to.
|
|
|
|
bool DownloadCovers(const std::vector<std::string>& url_templates, bool use_serial = false, ProgressCallback* progress = nullptr,
|
|
|
|
std::function<void(const Entry*, std::string)> save_callback = {});
|
2023-07-23 18:38:17 +00:00
|
|
|
|
|
|
|
// Custom properties support
|
|
|
|
void CheckCustomAttributesForPath(const std::string& path, bool& has_custom_title, bool& has_custom_region);
|
|
|
|
void SaveCustomTitleForPath(const std::string& path, const std::string& custom_title);
|
|
|
|
void SaveCustomRegionForPath(const std::string& path, int custom_region);
|
|
|
|
std::string GetCustomTitleForPath(const std::string& path);
|
2021-12-09 10:08:43 +00:00
|
|
|
} // namespace GameList
|
2023-10-02 04:57:20 +00:00
|
|
|
|
|
|
|
namespace Host
|
|
|
|
{
|
|
|
|
/// Called by Big Picture UI to begin cover download.
|
|
|
|
void OnCoverDownloaderOpenRequested();
|
|
|
|
}
|