2023-05-15 13:38:37 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2019-11-29 13:46:04 +00:00
|
|
|
#pragma once
|
2023-08-13 06:28:28 +00:00
|
|
|
|
|
|
|
#include "game_database.h"
|
|
|
|
#include "types.h"
|
|
|
|
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "util/cd_image.h"
|
2023-08-13 06:28:28 +00:00
|
|
|
|
2023-09-20 13:49:14 +00:00
|
|
|
#include "common/small_string.h"
|
2023-08-13 06:28:28 +00:00
|
|
|
|
2022-07-08 11:57:06 +00:00
|
|
|
#include <ctime>
|
2022-09-17 05:51:05 +00:00
|
|
|
#include <functional>
|
2022-07-11 13:03:29 +00:00
|
|
|
#include <mutex>
|
2023-09-05 11:07:20 +00:00
|
|
|
#include <span>
|
2019-11-29 13:46:04 +00:00
|
|
|
#include <string>
|
|
|
|
|
2020-01-08 03:37:43 +00:00
|
|
|
class ByteStream;
|
2020-03-12 05:32:19 +00:00
|
|
|
class ProgressCallback;
|
2019-11-29 13:46:04 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
struct SystemBootParameters;
|
2019-12-31 06:17:17 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
namespace GameList {
|
|
|
|
enum class EntryType
|
2019-11-29 13:46:04 +00:00
|
|
|
{
|
2020-01-24 04:50:44 +00:00
|
|
|
Disc,
|
2024-05-18 05:16:54 +00:00
|
|
|
DiscSet,
|
2020-08-15 09:54:33 +00:00
|
|
|
PSExe,
|
2021-01-24 04:06:52 +00:00
|
|
|
Playlist,
|
|
|
|
PSF,
|
|
|
|
Count
|
2020-01-24 04:50:44 +00:00
|
|
|
};
|
2019-12-04 11:12:50 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
struct Entry
|
2020-05-16 10:01:19 +00:00
|
|
|
{
|
2022-07-11 13:03:29 +00:00
|
|
|
EntryType type = EntryType::Disc;
|
2021-04-17 10:20:09 +00:00
|
|
|
DiscRegion region = DiscRegion::Other;
|
2020-01-24 04:50:46 +00:00
|
|
|
|
2020-01-24 04:50:44 +00:00
|
|
|
std::string path;
|
2022-07-11 13:03:29 +00:00
|
|
|
std::string serial;
|
2020-01-24 04:50:44 +00:00
|
|
|
std::string title;
|
2023-11-28 04:54:25 +00:00
|
|
|
std::string disc_set_name;
|
2021-04-17 04:23:47 +00:00
|
|
|
std::string genre;
|
|
|
|
std::string publisher;
|
|
|
|
std::string developer;
|
2023-05-15 13:38:37 +00:00
|
|
|
u64 hash = 0;
|
2023-12-20 13:40:24 +00:00
|
|
|
s64 file_size = 0;
|
|
|
|
u64 uncompressed_size = 0;
|
2022-07-11 13:03:29 +00:00
|
|
|
std::time_t last_modified_time = 0;
|
2022-10-21 11:02:19 +00:00
|
|
|
std::time_t last_played_time = 0;
|
|
|
|
std::time_t total_played_time = 0;
|
2021-04-17 04:23:47 +00:00
|
|
|
|
2021-04-17 10:20:09 +00:00
|
|
|
u64 release_date = 0;
|
2023-08-23 07:32:12 +00:00
|
|
|
u16 supported_controllers = static_cast<u16>(~0u);
|
2021-04-17 10:20:09 +00:00
|
|
|
u8 min_players = 1;
|
|
|
|
u8 max_players = 1;
|
|
|
|
u8 min_blocks = 0;
|
|
|
|
u8 max_blocks = 0;
|
2023-11-28 04:54:25 +00:00
|
|
|
s8 disc_set_index = -1;
|
2024-05-18 05:16:54 +00:00
|
|
|
bool disc_set_member = false;
|
2021-04-17 04:23:47 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
GameDatabase::CompatibilityRating compatibility = GameDatabase::CompatibilityRating::Unknown;
|
2021-04-17 04:23:47 +00:00
|
|
|
|
|
|
|
size_t GetReleaseDateString(char* buffer, size_t buffer_size) const;
|
2020-05-16 10:01:19 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
ALWAYS_INLINE bool IsDisc() const { return (type == EntryType::Disc); }
|
2024-05-18 05:16:54 +00:00
|
|
|
ALWAYS_INLINE bool IsDiscSet() const { return (type == EntryType::DiscSet); }
|
2024-05-23 02:42:50 +00:00
|
|
|
ALWAYS_INLINE EntryType GetSortType() const { return (type == EntryType::DiscSet) ? EntryType::Disc : type; }
|
2019-11-29 13:46:04 +00:00
|
|
|
};
|
2022-07-11 13:03:29 +00:00
|
|
|
|
|
|
|
const char* GetEntryTypeName(EntryType type);
|
|
|
|
const char* GetEntryTypeDisplayName(EntryType type);
|
|
|
|
|
2024-05-05 10:21:54 +00:00
|
|
|
bool IsScannableFilename(std::string_view path);
|
2022-07-11 13:03:29 +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, Entry* entry);
|
|
|
|
|
|
|
|
// 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);
|
2024-05-18 03:26:15 +00:00
|
|
|
const Entry* GetEntryForPath(std::string_view path);
|
2023-11-28 04:54:25 +00:00
|
|
|
const Entry* GetEntryBySerial(std::string_view serial);
|
|
|
|
const Entry* GetEntryBySerialAndHash(std::string_view serial, u64 hash);
|
2024-05-18 05:16:54 +00:00
|
|
|
std::vector<const Entry*> GetDiscSetMembers(std::string_view disc_set_name, bool sort_by_most_recent = false);
|
|
|
|
const Entry* GetFirstDiscSetMember(std::string_view disc_set_name);
|
2022-07-11 13:03:29 +00:00
|
|
|
u32 GetEntryCount();
|
|
|
|
|
|
|
|
bool IsGameListLoaded();
|
|
|
|
|
|
|
|
/// 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);
|
|
|
|
|
2022-10-21 11:02:19 +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);
|
2023-01-11 10:21:27 +00:00
|
|
|
void ClearPlayedTimeForSerial(const std::string& serial);
|
2022-10-21 11:02:19 +00:00
|
|
|
|
2022-10-23 04:09:54 +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);
|
|
|
|
|
2022-10-21 11:02:19 +00:00
|
|
|
/// Formats a timestamp to something human readable (e.g. Today, Yesterday, 10/11/12).
|
|
|
|
TinyString FormatTimestamp(std::time_t timestamp);
|
|
|
|
|
2022-10-23 04:09:54 +00:00
|
|
|
/// Formats a timespan to something human readable (e.g. 1h2m3s or 1 hour).
|
|
|
|
TinyString FormatTimespan(std::time_t timespan, bool long_format = false);
|
2022-10-21 11:02:19 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
std::string GetCoverImagePathForEntry(const Entry* entry);
|
|
|
|
std::string GetCoverImagePath(const std::string& path, const std::string& serial, const std::string& title);
|
2022-09-17 05:51:05 +00:00
|
|
|
std::string GetNewCoverImagePathForEntry(const Entry* entry, const char* new_filename, bool use_serial);
|
2022-09-09 10:32:21 +00:00
|
|
|
|
2023-08-23 08:12:10 +00:00
|
|
|
/// Returns a list of (title, entry) for entries matching serials. Titles will match the gamedb title,
|
|
|
|
/// except when two files have the same serial, in which case the filename will be used instead.
|
|
|
|
std::vector<std::pair<std::string, const Entry*>>
|
2023-09-05 11:07:20 +00:00
|
|
|
GetMatchingEntriesForSerial(const std::span<const std::string> serials);
|
2023-08-23 08:12:10 +00:00
|
|
|
|
2022-09-17 05:51:05 +00:00
|
|
|
/// 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 = {});
|
2022-07-11 13:03:29 +00:00
|
|
|
}; // namespace GameList
|
|
|
|
|
|
|
|
namespace Host {
|
|
|
|
/// Asynchronously starts refreshing the game list.
|
|
|
|
void RefreshGameListAsync(bool invalidate_cache);
|
|
|
|
|
|
|
|
/// Cancels game list refresh, if there is one in progress.
|
|
|
|
void CancelGameListRefresh();
|
2022-08-16 11:22:59 +00:00
|
|
|
} // namespace Host
|