Duckstation/src/frontend-common/game_list.h

113 lines
3.9 KiB
C
Raw Normal View History

// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin <stenzek@gmail.com>
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
#pragma once
2022-10-21 11:02:19 +00:00
#include "common/string.h"
#include "core/game_database.h"
#include "core/types.h"
#include "util/cd_image.h"
2022-07-08 11:57:06 +00:00
#include <ctime>
2022-09-17 05:51:05 +00:00
#include <functional>
#include <mutex>
#include <string>
2020-01-08 03:37:43 +00:00
class ByteStream;
class ProgressCallback;
struct SystemBootParameters;
2019-12-31 06:17:17 +00:00
namespace GameList {
enum class EntryType
{
Disc,
2020-08-15 09:54:33 +00:00
PSExe,
Playlist,
PSF,
Count
};
2019-12-04 11:12:50 +00:00
struct Entry
{
EntryType type = EntryType::Disc;
DiscRegion region = DiscRegion::Other;
std::string path;
std::string serial;
std::string title;
std::string genre;
std::string publisher;
std::string developer;
u64 total_size = 0;
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;
u64 release_date = 0;
u32 supported_controllers = ~static_cast<u32>(0);
u8 min_players = 1;
u8 max_players = 1;
u8 min_blocks = 0;
u8 max_blocks = 0;
GameDatabase::CompatibilityRating compatibility = GameDatabase::CompatibilityRating::Unknown;
size_t GetReleaseDateString(char* buffer, size_t buffer_size) const;
ALWAYS_INLINE bool IsDisc() const { return (type == EntryType::Disc); }
};
const char* GetEntryTypeName(EntryType type);
const char* GetEntryTypeDisplayName(EntryType type);
bool IsScannableFilename(const std::string_view& path);
/// 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);
const Entry* GetEntryForPath(const char* path);
const Entry* GetEntryBySerial(const std::string_view& serial);
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
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
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 = {});
}; // 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