OpenTTD Source 20260421-master-gc2fbc6fdeb
base_media_base.h
Go to the documentation of this file.
1/*
2 * This file is part of OpenTTD.
3 * OpenTTD 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.
4 * OpenTTD 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.
5 * See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenTTD. If not, see <https://www.gnu.org/licenses/old-licenses/gpl-2.0>.
6 */
7
9
10#ifndef BASE_MEDIA_BASE_H
11#define BASE_MEDIA_BASE_H
12
13#include "fileio_func.h"
14#include "textfile_type.h"
15#include "textfile_gui.h"
16#include "3rdparty/md5/md5.h"
17#include <unordered_map>
18
19struct IniFile;
20struct IniGroup;
21struct IniItem;
22struct ContentInfo;
23
25struct MD5File {
27 enum class ChecksumResult : uint8_t {
32 };
33
34 std::string filename;
35 MD5Hash hash;
36 std::string missing_warning;
38
39 ChecksumResult CheckMD5(Subdirectory subdir, size_t max_size) const;
40};
41
43template <class T> struct BaseSetTraits;
44
49template <class T>
50struct BaseSet {
52 virtual ~BaseSet() = default;
53
55 using TranslatedStrings = std::unordered_map<std::string, std::string, StringHash, std::equal_to<>>;
56
58 static constexpr size_t NUM_FILES = BaseSetTraits<T>::num_files;
59
62
64 static constexpr std::string_view SET_TYPE = BaseSetTraits<T>::set_type;
65
66 std::string name;
67 std::string url;
69 uint32_t shortname = 0;
70 std::vector<uint32_t> version;
71 bool fallback = false;
72
73 std::array<MD5File, BaseSet<T>::NUM_FILES> files{};
74 uint found_files = 0;
75 uint valid_files = 0;
76
81 int GetNumMissing() const
82 {
83 return BaseSet<T>::NUM_FILES - this->found_files;
84 }
85
91 int GetNumInvalid() const
92 {
93 return BaseSet<T>::NUM_FILES - this->valid_files;
94 }
95
96 void LogError(std::string_view full_filename, std::string_view detail, int level = 0) const;
97 const IniItem *GetMandatoryItem(std::string_view full_filename, const IniGroup &group, std::string_view name) const;
98
99 bool FillSetDetails(const IniFile &ini, const std::string &path, const std::string &full_filename, bool allow_empty_filename = true);
100
105 void CopyCompatibleConfig([[maybe_unused]] const T &src) {}
106
115 const std::string &GetDescription(std::string_view isocode) const
116 {
117 if (!isocode.empty()) {
118 /* First the full ISO code */
119 auto desc = this->description.find(isocode);
120 if (desc != this->description.end()) return desc->second;
121
122 /* Then the first two characters */
123 desc = this->description.find(isocode.substr(0, 2));
124 if (desc != this->description.end()) return desc->second;
125 }
126 /* Then fall back */
127 return this->description.at(std::string{});
128 }
129
140 {
141 return file->CheckMD5(subdir, SIZE_MAX);
142 }
143
149 std::optional<std::string> GetTextfile(TextfileType type) const
150 {
151 for (const auto &file : this->files) {
152 auto textfile = ::GetTextfile(type, Subdirectory::Baseset, file.filename);
153 if (textfile.has_value()) {
154 return textfile;
155 }
156 }
157 return std::nullopt;
158 }
159
164 static std::span<const std::string_view> GetFilenames();
165};
166
171template <class Tbase_set>
173protected:
174 static inline std::vector<std::unique_ptr<Tbase_set>> available_sets;
175 static inline std::vector<std::unique_ptr<Tbase_set>> duplicate_sets;
176 static inline const Tbase_set *used_set;
177
178 bool AddFile(const std::string &filename, size_t basepath_length, const std::string &tar_filename) override;
179
184 static std::string_view GetExtension();
185
190 static std::span<const std::unique_ptr<Tbase_set>> GetDuplicateSets() { return BaseMedia<Tbase_set>::duplicate_sets; }
191public:
197 static bool DetermineBestSet();
198
203 static uint FindSets()
204 {
206 /* Searching in tars is only done in the old "data" directories basesets. */
207 uint num = fs.Scan(GetExtension(), Tbase_set::SEARCH_IN_TARS ? Subdirectory::OldData : Subdirectory::OldGm, Tbase_set::SEARCH_IN_TARS);
208 return num + fs.Scan(GetExtension(), Subdirectory::Baseset, Tbase_set::SEARCH_IN_TARS);
209 }
210
215 static std::span<const std::unique_ptr<Tbase_set>> GetAvailableSets() { return BaseMedia<Tbase_set>::available_sets; }
216
217 static bool SetSet(const Tbase_set *set);
218 static bool SetSetByName(const std::string &name);
219 static bool SetSetByShortname(uint32_t shortname);
220 static void GetSetsList(std::back_insert_iterator<std::string> &output_iterator);
221 static int GetNumSets();
222 static int GetIndexOfUsedSet();
223 static const Tbase_set *GetSet(int index);
224 static const Tbase_set *GetUsedSet();
225
232 static bool HasSet(const ContentInfo &ci, bool md5sum);
233};
234
242template <class Tbase_set>
243std::optional<std::string_view> TryGetBaseSetFile(const ContentInfo &ci, bool md5sum, std::span<const std::unique_ptr<Tbase_set>> sets);
244
245#endif /* BASE_MEDIA_BASE_H */
std::optional< std::string_view > TryGetBaseSetFile(const ContentInfo &ci, bool md5sum, std::span< const std::unique_ptr< Tbase_set > > sets)
Check whether there's a base set matching some information.
Base for all base media (graphics, sounds).
static const Tbase_set * GetUsedSet()
Return the used set.
static std::vector< std::unique_ptr< Tbase_set > > available_sets
All available sets.
bool AddFile(const std::string &filename, size_t basepath_length, const std::string &tar_filename) override
Add a file with the given filename.
static bool SetSetByShortname(uint32_t shortname)
Set the set to be used.
static void GetSetsList(std::back_insert_iterator< std::string > &output_iterator)
Returns a list with the sets.
static std::span< const std::unique_ptr< Tbase_set > > GetAvailableSets()
Return the available sets.
static int GetNumSets()
Count the number of available graphics sets.
static uint FindSets()
Do the scan for files.
static bool HasSet(const ContentInfo &ci, bool md5sum)
Check whether we have an set with the exact characteristics as ci.
static const Tbase_set * used_set
The currently used set.
static std::vector< std::unique_ptr< Tbase_set > > duplicate_sets
All sets that aren't available, but needed for not downloading base sets when a newer version than th...
static const Tbase_set * GetSet(int index)
Get the base set at a specified index.
static int GetIndexOfUsedSet()
Get the index of the currently active graphics set.
static bool DetermineBestSet()
Determine the graphics pack that has to be used.
static bool SetSetByName(const std::string &name)
Set the set to be used.
static std::span< const std::unique_ptr< Tbase_set > > GetDuplicateSets()
Return the duplicate sets.
static bool SetSet(const Tbase_set *set)
Set the set to be used.
static std::string_view GetExtension()
Get the extension that is used to identify this set.
Helper for scanning for files with a given name.
Definition fileio_func.h:37
uint Scan(std::string_view extension, Subdirectory sd, bool tars=true, bool recursive=true)
Scan for files with the given extension in the given search path.
Definition fileio.cpp:1123
#define T
Climate temperate.
Definition engines.h:91
Functions for standard in/out file operations.
Subdirectory
The different kinds of subdirectories OpenTTD uses.
Definition fileio_type.h:88
@ Baseset
Subdirectory for all base data (base sets, intro game).
Definition fileio_type.h:96
@ OldData
Old subdirectory for the data.
Definition fileio_type.h:95
@ OldGm
Old subdirectory for the music.
Definition fileio_type.h:94
Defines the traits of a BaseSet type.
Information about a single base set.
uint found_files
Number of the files that could be found.
std::array< MD5File, BaseSet< T >::NUM_FILES > files
All files part of this set.
uint valid_files
Number of the files that could be found and are valid.
void LogError(std::string_view full_filename, std::string_view detail, int level=0) const
Log error from reading basesets.
const IniItem * GetMandatoryItem(std::string_view full_filename, const IniGroup &group, std::string_view name) const
Try to read a single piece of metadata and return nullptr if it doesn't exist.
std::string url
URL for information about the base set.
TranslatedStrings description
Description of the base set.
static constexpr size_t NUM_FILES
Number of files in this set.
std::optional< std::string > GetTextfile(TextfileType type) const
Search a textfile file next to this base media.
static constexpr bool SEARCH_IN_TARS
Whether to search in the tars or not.
static MD5File::ChecksumResult CheckMD5(const MD5File *file, Subdirectory subdir)
Calculate and check the MD5 hash of the supplied file.
void CopyCompatibleConfig(const T &src)
Copy settings from the given set into this set when they are compatible.
static std::span< const std::string_view > GetFilenames()
Get the internal names of the files in this set.
static constexpr std::string_view SET_TYPE
BaseSet type name.
std::string name
The name of the base set.
bool fallback
This set is a fallback set, i.e. it should be used only as last resort.
bool FillSetDetails(const IniFile &ini, const std::string &path, const std::string &full_filename, bool allow_empty_filename=true)
Read the set information from a loaded ini.
int GetNumMissing() const
Get the number of missing files.
std::vector< uint32_t > version
The version of this base set.
virtual ~BaseSet()=default
Ensure the destructor of the sub classes are called as well.
std::unordered_map< std::string, std::string, StringHash, std::equal_to<> > TranslatedStrings
Mapping of translations: language -> string.
uint32_t shortname
Four letter short variant of the name.
int GetNumInvalid() const
Get the number of invalid files.
const std::string & GetDescription(std::string_view isocode) const
Get the description for the given ISO code.
Container for all important information about a piece of content.
Ini file that supports both loading and saving.
Definition ini_type.h:87
A group within an ini file.
Definition ini_type.h:34
A single "line" in an ini file.
Definition ini_type.h:23
Structure holding filename and MD5 information about a single file.
std::string missing_warning
warning when this file is missing
ChecksumResult check_result
cached result of md5 check
ChecksumResult CheckMD5(Subdirectory subdir, size_t max_size) const
Calculate and check the MD5 hash of the supplied filename.
Definition gfxinit.cpp:447
ChecksumResult
The result of a checksum check.
@ NoFile
The file did not exist.
@ Mismatch
The file did exist, just the md5 checksum did not match.
@ Match
The file did exist and the md5 checksum did match.
@ Unknown
The file has not been checked yet.
MD5Hash hash
md5 sum of the file
std::string filename
filename
GUI functions related to textfiles.
Types related to textfiles.
TextfileType
Additional text files accompanying Tar archives.