blob: 744a8533eb53cf7881917609b4c23a43abbfbbaa [file] [log] [blame]
/*
Copyright 2012 to 2020 TeamWin
This file is part of TWRP/TeamWin Recovery Project.
TWRP 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, either version 3 of the License, or
(at your option) any later version.
TWRP 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 TWRP. If not, see <http://www.gnu.org/licenses/>.
*/
// resource.cpp - Source to manage GUI resources
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <string>
#include <sstream>
#include <iostream>
#include <iomanip>
#include <fcntl.h>
#include <ziparchive/zip_archive.h>
#include <android-base/unique_fd.h>
extern "C" {
#include "../twcommon.h"
#include "gui.h"
}
#include "minuitwrp/truetype.hpp"
#include "minuitwrp/minui.h"
#include "rapidxml.hpp"
#include "objects.hpp"
#define TMP_RESOURCE_NAME "/tmp/extract.bin"
Resource::Resource(xml_node<>* node, ZipArchiveHandle pZip __unused)
{
if (node && node->first_attribute("name"))
mName = node->first_attribute("name")->value();
}
int Resource::ExtractResource(ZipArchiveHandle pZip, std::string folderName, std::string fileName, std::string fileExtn, std::string destFile)
{
if (!pZip)
return -1;
std::string src = folderName + "/" + fileName + fileExtn;
ZipEntry binary_entry;
if (FindEntry(pZip, src, &binary_entry) == 0) {
android::base::unique_fd fd(
open(destFile.c_str(), O_CREAT | O_WRONLY | O_TRUNC | O_CLOEXEC, 0666));
if (fd == -1) {
return -1;
}
int32_t err = ExtractEntryToFile(pZip, &binary_entry, fd);
if (err != 0)
return -1;
} else {
return -1;
}
return 0;
}
void Resource::LoadImage(ZipArchiveHandle pZip, std::string file, gr_surface* surface)
{
int rc = 0;
if (ExtractResource(pZip, "images", file, ".png", TMP_RESOURCE_NAME) == 0)
{
rc = res_create_surface(TMP_RESOURCE_NAME, surface);
unlink(TMP_RESOURCE_NAME);
}
else if (ExtractResource(pZip, "images", file, "", TMP_RESOURCE_NAME) == 0)
{
// JPG includes the .jpg extension in the filename so extension should be blank
rc = res_create_surface(TMP_RESOURCE_NAME, surface);
unlink(TMP_RESOURCE_NAME);
}
else if (!pZip)
{
// File name in xml may have included .png so try without adding .png
rc = res_create_surface(file.c_str(), surface);
}
if (rc != 0)
LOGINFO("Failed to load image from %s%s, error %d\n", file.c_str(), pZip ? " (zip)" : "", rc);
}
void Resource::CheckAndScaleImage(gr_surface source, gr_surface* destination, int retain_aspect)
{
if (!source) {
*destination = nullptr;
return;
}
if (get_scale_w() != 0 && get_scale_h() != 0) {
float scale_w = get_scale_w(), scale_h = get_scale_h();
if (retain_aspect) {
if (scale_w < scale_h)
scale_h = scale_w;
else
scale_w = scale_h;
}
if (res_scale_surface(source, destination, scale_w, scale_h)) {
LOGINFO("Error scaling image, using regular size.\n");
*destination = source;
}
} else {
*destination = source;
}
}
FontResource::FontResource(xml_node<>* node, ZipArchiveHandle pZip)
: Resource(node, pZip)
{
origFontSize = 0;
origFont = NULL;
LoadFont(node, pZip);
}
void FontResource::LoadFont(xml_node<>* node, ZipArchiveHandle pZip)
{
std::string file;
xml_attribute<>* attr;
mFont = NULL;
if (!node)
return;
attr = node->first_attribute("filename");
if (!attr)
return;
file = attr->value();
if (file.size() >= 4 && file.compare(file.size()-4, 4, ".ttf") == 0)
{
int font_size = 0;
if (origFontSize != 0) {
attr = node->first_attribute("scale");
if (attr == NULL)
return;
font_size = origFontSize * atoi(attr->value()) / 100;
} else {
attr = node->first_attribute("size");
if (attr == NULL)
return;
font_size = scale_theme_min(atoi(attr->value()));
origFontSize = font_size;
}
int dpi = 300;
attr = node->first_attribute("dpi");
if (attr)
dpi = atoi(attr->value());
// we can't use TMP_RESOURCE_NAME here because the ttf subsystem is caching the name and scaling needs to reload the font
std::string tmpname = "/tmp/" + file;
if (ExtractResource(pZip, "fonts", file, "", tmpname) == 0)
{
mFont = twrpTruetype::gr_ttf_loadFont(tmpname.c_str(), font_size, dpi);
}
else
{
file = std::string(TWRES "fonts/") + file;
mFont = twrpTruetype::gr_ttf_loadFont(file.c_str(), font_size, dpi);
}
}
else
{
LOGERR("Non-TTF fonts are no longer supported.\n");
}
}
void FontResource::DeleteFont() {
if (mFont) {
twrpTruetype::gr_ttf_freeFont(mFont);
}
mFont = NULL;
if (origFont) {
twrpTruetype::gr_ttf_freeFont(origFont);
}
origFont = NULL;
}
void FontResource::Override(xml_node<>* node, ZipArchiveHandle pZip) {
if (!origFont) {
origFont = mFont;
} else if (mFont) {
twrpTruetype::gr_ttf_freeFont(mFont);
mFont = NULL;
}
LoadFont(node, pZip);
}
FontResource::~FontResource()
{
DeleteFont();
}
ImageResource::ImageResource(xml_node<>* node, ZipArchiveHandle pZip)
: Resource(node, pZip)
{
std::string file;
gr_surface temp_surface = nullptr;
mSurface = NULL;
if (!node) {
LOGERR("ImageResource node is NULL\n");
return;
}
if (node->first_attribute("filename"))
file = node->first_attribute("filename")->value();
else {
LOGERR("No filename specified for image resource.\n");
return;
}
bool retain_aspect = (node->first_attribute("retainaspect") != NULL);
// the value does not matter, if retainaspect is present, we assume that we want to retain it
LoadImage(pZip, file, &temp_surface);
CheckAndScaleImage(temp_surface, &mSurface, retain_aspect);
}
ImageResource::~ImageResource()
{
if (mSurface)
res_free_surface(mSurface);
}
AnimationResource::AnimationResource(xml_node<>* node, ZipArchiveHandle pZip)
: Resource(node, pZip)
{
std::string file;
int fileNum = 1;
if (!node)
return;
if (node->first_attribute("filename"))
file = node->first_attribute("filename")->value();
else {
LOGERR("No filename specified for image resource.\n");
return;
}
bool retain_aspect = (node->first_attribute("retainaspect") != NULL);
// the value does not matter, if retainaspect is present, we assume that we want to retain it
for (;;)
{
std::ostringstream fileName;
fileName << file << std::setfill ('0') << std::setw (3) << fileNum;
gr_surface surface = nullptr;
gr_surface temp_surface = nullptr;
LoadImage(pZip, fileName.str(), &temp_surface);
CheckAndScaleImage(temp_surface, &surface, retain_aspect);
if (surface) {
mSurfaces.push_back(surface);
fileNum++;
} else
break; // Done loading animation images
}
}
AnimationResource::~AnimationResource()
{
std::vector<gr_surface>::iterator it;
for (it = mSurfaces.begin(); it != mSurfaces.end(); ++it)
res_free_surface(*it);
mSurfaces.clear();
}
FontResource* ResourceManager::FindFont(const std::string& name) const
{
for (std::vector<FontResource*>::const_iterator it = mFonts.begin(); it != mFonts.end(); ++it)
if (name == (*it)->GetName())
return *it;
return NULL;
}
ImageResource* ResourceManager::FindImage(const std::string& name) const
{
for (std::vector<ImageResource*>::const_iterator it = mImages.begin(); it != mImages.end(); ++it)
if (name == (*it)->GetName())
return *it;
return NULL;
}
AnimationResource* ResourceManager::FindAnimation(const std::string& name) const
{
for (std::vector<AnimationResource*>::const_iterator it = mAnimations.begin(); it != mAnimations.end(); ++it)
if (name == (*it)->GetName())
return *it;
return NULL;
}
std::string ResourceManager::FindString(const std::string& name) const
{
//if (this != NULL) {
std::map<std::string, string_resource_struct>::const_iterator it = mStrings.find(name);
if (it != mStrings.end())
return it->second.value;
LOGERR("String resource '%s' not found. No default value.\n", name.c_str());
PageManager::AddStringResource("NO DEFAULT", name, "[" + name + ("]"));
/*} else {
LOGINFO("String resources not loaded when looking for '%s'. No default value.\n", name.c_str());
}*/
return "[" + name + ("]");
}
std::string ResourceManager::FindString(const std::string& name, const std::string& default_string) const
{
//if (this != NULL) {
std::map<std::string, string_resource_struct>::const_iterator it = mStrings.find(name);
if (it != mStrings.end())
return it->second.value;
LOGERR("String resource '%s' not found. Using default value.\n", name.c_str());
PageManager::AddStringResource("DEFAULT", name, default_string);
/*} else {
LOGINFO("String resources not loaded when looking for '%s'. Using default value.\n", name.c_str());
}*/
return default_string;
}
void ResourceManager::DumpStrings() const
{
/*if (this == NULL) {
gui_print("No string resources\n");
return;
}*/
std::map<std::string, string_resource_struct>::const_iterator it;
gui_print("Dumping all strings:\n");
for (it = mStrings.begin(); it != mStrings.end(); it++)
gui_print("source: %s: '%s' = '%s'\n", it->second.source.c_str(), it->first.c_str(), it->second.value.c_str());
gui_print("Done dumping strings\n");
}
ResourceManager::ResourceManager()
{
}
void ResourceManager::AddStringResource(std::string resource_source, std::string resource_name, std::string value)
{
string_resource_struct res;
res.source = resource_source;
res.value = value;
mStrings[resource_name] = res;
}
void ResourceManager::LoadResources(xml_node<>* resList, ZipArchiveHandle pZip, std::string resource_source)
{
if (!resList)
return;
for (xml_node<>* child = resList->first_node(); child; child = child->next_sibling())
{
std::string type = child->name();
if (type == "resource") {
// legacy format : <resource type="...">
xml_attribute<>* attr = child->first_attribute("type");
type = attr ? attr->value() : "*unspecified*";
}
bool error = false;
if (type == "font")
{
FontResource* res = new FontResource(child, pZip);
if (res && res->GetResource())
mFonts.push_back(res);
else {
error = true;
delete res;
}
}
else if (type == "fontoverride")
{
if (mFonts.size() != 0 && child && child->first_attribute("name")) {
string FontName = child->first_attribute("name")->value();
size_t font_count = mFonts.size(), i;
bool found = false;
for (i = 0; i < font_count; i++) {
if (mFonts[i]->GetName() == FontName) {
mFonts[i]->Override(child, pZip);
found = true;
break;
}
}
if (!found) {
LOGERR("Unable to locate font '%s' for override.\n", FontName.c_str());
}
} else if (mFonts.size() != 0)
LOGERR("Unable to locate font name for type fontoverride.\n");
}
else if (type == "image")
{
ImageResource* res = new ImageResource(child, pZip);
if (res && res->GetResource())
mImages.push_back(res);
else {
error = true;
delete res;
}
}
else if (type == "animation")
{
AnimationResource* res = new AnimationResource(child, pZip);
if (res && res->GetResourceCount())
mAnimations.push_back(res);
else {
error = true;
delete res;
}
}
else if (type == "string")
{
if (xml_attribute<>* attr = child->first_attribute("name")) {
string_resource_struct res;
res.source = resource_source;
res.value = child->value();
mStrings[attr->value()] = res;
} else
error = true;
}
else
{
LOGERR("Resource type (%s) not supported.\n", type.c_str());
error = true;
}
if (error)
{
std::string res_name;
if (child->first_attribute("name"))
res_name = child->first_attribute("name")->value();
if (res_name.empty() && child->first_attribute("filename"))
res_name = child->first_attribute("filename")->value();
if (!res_name.empty()) {
LOGERR("Resource (%s)-(%s) failed to load\n", type.c_str(), res_name.c_str());
} else
LOGERR("Resource type (%s) failed to load\n", type.c_str());
}
}
}
ResourceManager::~ResourceManager()
{
for (std::vector<FontResource*>::iterator it = mFonts.begin(); it != mFonts.end(); ++it)
delete *it;
for (std::vector<ImageResource*>::iterator it = mImages.begin(); it != mImages.end(); ++it)
delete *it;
for (std::vector<AnimationResource*>::iterator it = mAnimations.begin(); it != mAnimations.end(); ++it)
delete *it;
}