2018-11-17 15:36:28 +00:00
|
|
|
// Copyright 2018 Dolphin Emulator Project
|
2021-07-05 01:22:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2018-11-17 15:36:28 +00:00
|
|
|
|
|
|
|
#include "UICommon/ResourcePack/ResourcePack.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
2022-08-06 20:26:04 +00:00
|
|
|
#include <memory>
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2022-05-28 22:19:55 +00:00
|
|
|
#include <mz_compat.h>
|
2022-08-06 20:26:04 +00:00
|
|
|
#include <mz_os.h>
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2019-10-06 06:21:06 +00:00
|
|
|
#include "Common/CommonPaths.h"
|
2024-09-21 05:17:29 +00:00
|
|
|
#include "Common/Contains.h"
|
2018-11-17 15:36:28 +00:00
|
|
|
#include "Common/FileSearch.h"
|
|
|
|
#include "Common/FileUtil.h"
|
2022-08-06 20:26:04 +00:00
|
|
|
#include "Common/IOFile.h"
|
2019-08-23 20:38:44 +00:00
|
|
|
#include "Common/MinizipUtil.h"
|
2019-05-27 17:02:01 +00:00
|
|
|
#include "Common/ScopeGuard.h"
|
2018-11-17 15:36:28 +00:00
|
|
|
#include "Common/StringUtil.h"
|
|
|
|
|
|
|
|
#include "UICommon/ResourcePack/Manager.h"
|
|
|
|
#include "UICommon/ResourcePack/Manifest.h"
|
|
|
|
|
|
|
|
namespace ResourcePack
|
|
|
|
{
|
2019-10-06 06:21:06 +00:00
|
|
|
constexpr char TEXTURE_PATH[] = HIRES_TEXTURES_DIR DIR_SEP;
|
2019-05-27 16:33:46 +00:00
|
|
|
|
2018-11-17 15:36:28 +00:00
|
|
|
ResourcePack::ResourcePack(const std::string& path) : m_path(path)
|
|
|
|
{
|
|
|
|
auto file = unzOpen(path.c_str());
|
2019-05-27 17:02:01 +00:00
|
|
|
Common::ScopeGuard file_guard{[&] { unzClose(file); }};
|
2018-11-17 15:36:28 +00:00
|
|
|
|
|
|
|
if (file == nullptr)
|
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Failed to open resource pack";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-04-22 07:14:34 +00:00
|
|
|
if (unzLocateFile(file, "manifest.json", 0) == UNZ_END_OF_LIST_OF_FILE)
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Resource pack is missing a manifest.";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-08-06 20:26:04 +00:00
|
|
|
unz_file_info64 manifest_info{};
|
|
|
|
unzGetCurrentFileInfo64(file, &manifest_info, nullptr, 0, nullptr, 0, nullptr, 0);
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2019-05-27 17:27:03 +00:00
|
|
|
std::string manifest_contents(manifest_info.uncompressed_size, '\0');
|
2019-08-23 20:38:44 +00:00
|
|
|
if (!Common::ReadFileFromZip(file, &manifest_contents))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Failed to read manifest.json";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
unzCloseCurrentFile(file);
|
|
|
|
|
2019-05-27 17:27:03 +00:00
|
|
|
m_manifest = std::make_shared<Manifest>(manifest_contents);
|
2018-11-17 15:36:28 +00:00
|
|
|
if (!m_manifest->IsValid())
|
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Manifest error: " + m_manifest->GetError();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-04-22 07:14:34 +00:00
|
|
|
if (unzLocateFile(file, "logo.png", 0) != UNZ_END_OF_LIST_OF_FILE)
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
2022-08-06 20:26:04 +00:00
|
|
|
unz_file_info64 logo_info{};
|
|
|
|
unzGetCurrentFileInfo64(file, &logo_info, nullptr, 0, nullptr, 0, nullptr, 0);
|
2018-11-17 15:36:28 +00:00
|
|
|
|
|
|
|
m_logo_data.resize(logo_info.uncompressed_size);
|
|
|
|
|
2019-08-23 20:38:44 +00:00
|
|
|
if (!Common::ReadFileFromZip(file, &m_logo_data))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Failed to read logo.png";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unzGoToFirstFile(file);
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
2019-05-27 16:37:43 +00:00
|
|
|
std::string filename(256, '\0');
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2022-08-06 20:26:04 +00:00
|
|
|
unz_file_info64 texture_info{};
|
|
|
|
unzGetCurrentFileInfo64(file, &texture_info, filename.data(), static_cast<u16>(filename.size()),
|
|
|
|
nullptr, 0, nullptr, 0);
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2024-04-19 18:24:34 +00:00
|
|
|
if (!filename.starts_with("textures/") || texture_info.uncompressed_size == 0)
|
2018-11-17 15:36:28 +00:00
|
|
|
continue;
|
|
|
|
|
2019-02-02 14:54:06 +00:00
|
|
|
// If a texture is compressed and the manifest doesn't state that, abort.
|
|
|
|
if (!m_manifest->IsCompressed() && texture_info.compression_method != 0)
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Texture " + filename + " is compressed!";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_textures.push_back(filename.substr(9));
|
|
|
|
} while (unzGoToNextFile(file) != UNZ_END_OF_LIST_OF_FILE);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ResourcePack::IsValid() const
|
|
|
|
{
|
|
|
|
return m_valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::vector<char>& ResourcePack::GetLogo() const
|
|
|
|
{
|
|
|
|
return m_logo_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& ResourcePack::GetPath() const
|
|
|
|
{
|
|
|
|
return m_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& ResourcePack::GetError() const
|
|
|
|
{
|
|
|
|
return m_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Manifest* ResourcePack::GetManifest() const
|
|
|
|
{
|
|
|
|
return m_manifest.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::vector<std::string>& ResourcePack::GetTextures() const
|
|
|
|
{
|
|
|
|
return m_textures;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ResourcePack::Install(const std::string& path)
|
|
|
|
{
|
|
|
|
if (!IsValid())
|
|
|
|
{
|
|
|
|
m_error = "Invalid pack";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto file = unzOpen(m_path.c_str());
|
2022-08-06 20:26:04 +00:00
|
|
|
if (file == nullptr)
|
|
|
|
{
|
|
|
|
m_valid = false;
|
|
|
|
m_error = "Failed to open resource pack";
|
|
|
|
return false;
|
|
|
|
}
|
2019-05-27 17:02:01 +00:00
|
|
|
Common::ScopeGuard file_guard{[&] { unzClose(file); }};
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2022-08-06 20:26:04 +00:00
|
|
|
if (unzGoToFirstFile(file) != MZ_OK)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::string texture_zip_path;
|
|
|
|
do
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
2022-08-06 20:26:04 +00:00
|
|
|
texture_zip_path.resize(UINT16_MAX + 1, '\0');
|
|
|
|
unz_file_info64 texture_info{};
|
|
|
|
if (unzGetCurrentFileInfo64(file, &texture_info, texture_zip_path.data(), UINT16_MAX, nullptr,
|
|
|
|
0, nullptr, 0) != MZ_OK)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
TruncateToCString(&texture_zip_path);
|
|
|
|
|
|
|
|
const std::string texture_zip_path_prefix = "textures/";
|
|
|
|
if (!texture_zip_path.starts_with(texture_zip_path_prefix))
|
|
|
|
continue;
|
|
|
|
const std::string texture_name = texture_zip_path.substr(texture_zip_path_prefix.size());
|
|
|
|
|
2024-09-22 01:09:34 +00:00
|
|
|
auto texture_it = std::ranges::find_if(m_textures, [&texture_name](const std::string& texture) {
|
|
|
|
return mz_path_compare_wc(texture.c_str(), texture_name.c_str(), 1) == MZ_OK;
|
|
|
|
});
|
2022-08-06 20:26:04 +00:00
|
|
|
if (texture_it == m_textures.cend())
|
|
|
|
continue;
|
|
|
|
const auto texture = *texture_it;
|
2018-11-17 15:36:28 +00:00
|
|
|
|
|
|
|
// Check if a higher priority pack already provides a given texture, don't overwrite it
|
2022-08-06 20:26:04 +00:00
|
|
|
bool provided_by_other_pack = false;
|
2018-11-17 15:36:28 +00:00
|
|
|
for (const auto& pack : GetHigherPriorityPacks(*this))
|
|
|
|
{
|
2024-09-21 05:17:29 +00:00
|
|
|
if (Common::Contains(pack->GetTextures(), texture))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
provided_by_other_pack = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (provided_by_other_pack)
|
|
|
|
continue;
|
|
|
|
|
2019-05-27 16:45:21 +00:00
|
|
|
const std::string texture_path = path + TEXTURE_PATH + texture;
|
2022-08-06 20:26:04 +00:00
|
|
|
std::string texture_full_dir;
|
|
|
|
if (!SplitPath(texture_path, &texture_full_dir, nullptr, nullptr))
|
|
|
|
continue;
|
2018-11-17 15:36:28 +00:00
|
|
|
|
2022-08-06 20:26:04 +00:00
|
|
|
if (!File::CreateFullPath(texture_full_dir))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
2022-08-06 20:26:04 +00:00
|
|
|
m_error = "Failed to create full path " + texture_full_dir;
|
2018-11-17 15:36:28 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-08-06 20:26:04 +00:00
|
|
|
const size_t data_size = static_cast<size_t>(texture_info.uncompressed_size);
|
|
|
|
auto data = std::make_unique<u8[]>(data_size);
|
|
|
|
if (!Common::ReadFileFromZip(file, data.get(), data_size))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_error = "Failed to read texture " + texture;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-08-06 20:26:04 +00:00
|
|
|
File::IOFile out(texture_path, "wb");
|
|
|
|
if (!out)
|
|
|
|
{
|
|
|
|
m_error = "Failed to open " + texture;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!out.WriteBytes(data.get(), data_size))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_error = "Failed to write " + texture;
|
|
|
|
return false;
|
|
|
|
}
|
2022-08-06 20:26:04 +00:00
|
|
|
} while (unzGoToNextFile(file) == MZ_OK);
|
2018-11-17 15:36:28 +00:00
|
|
|
|
|
|
|
SetInstalled(*this, true);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ResourcePack::Uninstall(const std::string& path)
|
|
|
|
{
|
|
|
|
if (!IsValid())
|
|
|
|
{
|
|
|
|
m_error = "Invalid pack";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto lower = GetLowerPriorityPacks(*this);
|
|
|
|
|
|
|
|
SetInstalled(*this, false);
|
|
|
|
|
|
|
|
for (const auto& texture : m_textures)
|
|
|
|
{
|
|
|
|
bool provided_by_other_pack = false;
|
|
|
|
|
|
|
|
// Check if a higher priority pack already provides a given texture, don't delete it
|
|
|
|
for (const auto& pack : GetHigherPriorityPacks(*this))
|
|
|
|
{
|
2024-09-21 05:17:29 +00:00
|
|
|
if (::ResourcePack::IsInstalled(*pack) && Common::Contains(pack->GetTextures(), texture))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
provided_by_other_pack = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (provided_by_other_pack)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Check if a lower priority pack provides a given texture - if so, install it.
|
|
|
|
for (auto& pack : lower)
|
|
|
|
{
|
2024-09-21 05:17:29 +00:00
|
|
|
if (::ResourcePack::IsInstalled(*pack) && Common::Contains(pack->GetTextures(), texture))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
pack->Install(path);
|
|
|
|
|
|
|
|
provided_by_other_pack = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (provided_by_other_pack)
|
|
|
|
continue;
|
|
|
|
|
2019-05-27 16:45:21 +00:00
|
|
|
const std::string texture_path = path + TEXTURE_PATH + texture;
|
|
|
|
if (File::Exists(texture_path) && !File::Delete(texture_path))
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
m_error = "Failed to delete texture " + texture;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recursively delete empty directories
|
|
|
|
|
|
|
|
std::string dir;
|
2019-05-27 16:45:21 +00:00
|
|
|
SplitPath(texture_path, &dir, nullptr, nullptr);
|
2018-11-17 15:36:28 +00:00
|
|
|
|
|
|
|
while (dir.length() > (path + TEXTURE_PATH).length())
|
|
|
|
{
|
|
|
|
auto is_empty = Common::DoFileSearch({dir}).empty();
|
|
|
|
|
|
|
|
if (is_empty)
|
|
|
|
File::DeleteDir(dir);
|
|
|
|
|
|
|
|
SplitPath(dir.substr(0, dir.size() - 2), &dir, nullptr, nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-03-13 19:53:30 +00:00
|
|
|
bool ResourcePack::operator==(const ResourcePack& pack) const
|
2018-11-17 15:36:28 +00:00
|
|
|
{
|
|
|
|
return pack.GetPath() == m_path;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace ResourcePack
|