mirror of
https://github.com/aseprite/aseprite.git
synced 2025-01-27 06:35:16 +00:00
Update to tinyxml2
This commit is contained in:
parent
4d18200b0c
commit
e48e71ba28
3
.gitmodules
vendored
3
.gitmodules
vendored
@ -78,3 +78,6 @@
|
|||||||
[submodule "third_party/qoi"]
|
[submodule "third_party/qoi"]
|
||||||
path = third_party/qoi
|
path = third_party/qoi
|
||||||
url = https://github.com/aseprite/qoi.git
|
url = https://github.com/aseprite/qoi.git
|
||||||
|
[submodule "third_party/tinyxml2"]
|
||||||
|
path = third_party/tinyxml2
|
||||||
|
url = https://github.com/aseprite/tinyxml2.git
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
# Aseprite
|
# Aseprite
|
||||||
# Copyright (C) 2018-2022 Igara Studio S.A.
|
# Copyright (C) 2018-2024 Igara Studio S.A.
|
||||||
# Copyright (C) 2001-2018 David Capello
|
# Copyright (C) 2001-2018 David Capello
|
||||||
|
|
||||||
cmake_minimum_required(VERSION 3.16)
|
cmake_minimum_required(VERSION 3.16)
|
||||||
@ -163,7 +163,7 @@ set(PIXMAN_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/pixman)
|
|||||||
set(FREETYPE_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/freetype2)
|
set(FREETYPE_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/freetype2)
|
||||||
set(HARFBUZZ_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/harfbuzz)
|
set(HARFBUZZ_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/harfbuzz)
|
||||||
set(SIMPLEINI_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/simpleini)
|
set(SIMPLEINI_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/simpleini)
|
||||||
set(TINYXML_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/tinyxml)
|
set(TINYXML_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/tinyxml2)
|
||||||
set(ZLIB_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/zlib)
|
set(ZLIB_DIR ${CMAKE_CURRENT_SOURCE_DIR}/third_party/zlib)
|
||||||
|
|
||||||
# Search in the "cmake" directory for additional CMake modules.
|
# Search in the "cmake" directory for additional CMake modules.
|
||||||
@ -225,12 +225,12 @@ endif()
|
|||||||
include_directories(${PNG_INCLUDE_DIRS})
|
include_directories(${PNG_INCLUDE_DIRS})
|
||||||
add_definitions(-DPNG_NO_MMX_CODE) # Do not use MMX optimizations in PNG code
|
add_definitions(-DPNG_NO_MMX_CODE) # Do not use MMX optimizations in PNG code
|
||||||
|
|
||||||
# tinyxml
|
# tinyxml2
|
||||||
if(USE_SHARED_TINYXML)
|
if(USE_SHARED_TINYXML)
|
||||||
find_library(TINYXML_LIBRARY NAMES tinyxml)
|
find_library(TINYXML_LIBRARY NAMES tinyxml2)
|
||||||
find_path(TINYXML_INCLUDE_DIR NAMES tinyxml.h)
|
find_path(TINYXML_INCLUDE_DIR NAMES tinyxml2.h)
|
||||||
else()
|
else()
|
||||||
set(TINYXML_LIBRARY tinyxml)
|
set(TINYXML_LIBRARY tinyxml2)
|
||||||
set(TINYXML_INCLUDE_DIR ${TINYXML_DIR})
|
set(TINYXML_INCLUDE_DIR ${TINYXML_DIR})
|
||||||
endif()
|
endif()
|
||||||
include_directories(${TINYXML_INCLUDE_DIR})
|
include_directories(${TINYXML_INCLUDE_DIR})
|
||||||
|
@ -1139,10 +1139,10 @@ freely, subject to the following restrictions:
|
|||||||
3. This notice may not be removed or altered from any source distribution.
|
3. This notice may not be removed or altered from any source distribution.
|
||||||
```
|
```
|
||||||
|
|
||||||
# [tinyxml](http://www.grinninglizard.com/tinyxml/)
|
# [tinyxml2](https://github.com/leethomason/tinyxml2)
|
||||||
|
|
||||||
```
|
```
|
||||||
TinyXML is released under the zlib license:
|
Original code by Lee Thomason (www.grinninglizard.com)
|
||||||
|
|
||||||
This software is provided 'as-is', without any express or implied
|
This software is provided 'as-is', without any express or implied
|
||||||
warranty. In no event will the authors be held liable for any
|
warranty. In no event will the authors be held liable for any
|
||||||
@ -1162,12 +1162,6 @@ must not be misrepresented as being the original software.
|
|||||||
|
|
||||||
3. This notice may not be removed or altered from any source
|
3. This notice may not be removed or altered from any source
|
||||||
distribution.
|
distribution.
|
||||||
|
|
||||||
--
|
|
||||||
|
|
||||||
TinyXML was originally written by Lee Thomason. Lee reviews changes
|
|
||||||
and releases new versions, with the help of Yves Berquin, Andrew
|
|
||||||
Ellerton, and the tinyXml community.
|
|
||||||
```
|
```
|
||||||
|
|
||||||
# [ucdn](https://github.com/grigorig/ucdn)
|
# [ucdn](https://github.com/grigorig/ucdn)
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2020-2023 Igara Studio S.A.
|
// Copyright (C) 2020-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2016 David Capello
|
// Copyright (C) 2001-2016 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -23,6 +23,8 @@
|
|||||||
#include "doc/image.h"
|
#include "doc/image.h"
|
||||||
#include "doc/image_impl.h"
|
#include "doc/image_impl.h"
|
||||||
|
|
||||||
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
@ -30,15 +32,16 @@ namespace app {
|
|||||||
using namespace doc;
|
using namespace doc;
|
||||||
using namespace base::serialization;
|
using namespace base::serialization;
|
||||||
using namespace base::serialization::little_endian;
|
using namespace base::serialization::little_endian;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
ImageRef load_xml_image(const TiXmlElement* imageElem)
|
ImageRef load_xml_image(const XMLElement* imageElem)
|
||||||
{
|
{
|
||||||
ImageRef image;
|
ImageRef image;
|
||||||
int w, h;
|
int w, h;
|
||||||
if (imageElem->QueryIntAttribute("width", &w) != TIXML_SUCCESS ||
|
if (imageElem->QueryIntAttribute("width", &w) != XML_SUCCESS ||
|
||||||
imageElem->QueryIntAttribute("height", &h) != TIXML_SUCCESS ||
|
imageElem->QueryIntAttribute("height", &h) != XML_SUCCESS ||
|
||||||
w < 0 || w > 9999 ||
|
w < 0 || w > 9999 ||
|
||||||
h < 0 || h > 9999)
|
h < 0 || h > 9999)
|
||||||
return image;
|
return image;
|
||||||
@ -109,7 +112,7 @@ ImageRef load_xml_image(const TiXmlElement* imageElem)
|
|||||||
return image;
|
return image;
|
||||||
}
|
}
|
||||||
|
|
||||||
void save_xml_image(TiXmlElement* imageElem, const Image* image)
|
void save_xml_image(XMLElement* imageElem, const Image* image)
|
||||||
{
|
{
|
||||||
int w = image->width();
|
int w = image->width();
|
||||||
int h = image->height();
|
int h = image->height();
|
||||||
@ -167,8 +170,7 @@ void save_xml_image(TiXmlElement* imageElem, const Image* image)
|
|||||||
|
|
||||||
std::string data_base64;
|
std::string data_base64;
|
||||||
base::encode_base64(data, data_base64);
|
base::encode_base64(data, data_base64);
|
||||||
TiXmlText textElem(data_base64.c_str());
|
imageElem->InsertNewText(data_base64.c_str());
|
||||||
imageElem->InsertEndChild(textElem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // anonymous namespace
|
} // anonymous namespace
|
||||||
@ -305,11 +307,11 @@ static const int kBrushFlags =
|
|||||||
|
|
||||||
void AppBrushes::load(const std::string& filename)
|
void AppBrushes::load(const std::string& filename)
|
||||||
{
|
{
|
||||||
XmlDocumentRef doc = app::open_xml(filename);
|
XMLDocumentRef doc = app::open_xml(filename);
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
TiXmlElement* brushElem = handle
|
XMLElement* brushElem = handle
|
||||||
.FirstChild("brushes")
|
.FirstChildElement("brushes")
|
||||||
.FirstChild("brush").ToElement();
|
.FirstChildElement("brush").ToElement();
|
||||||
|
|
||||||
while (brushElem) {
|
while (brushElem) {
|
||||||
// flags
|
// flags
|
||||||
@ -339,9 +341,9 @@ void AppBrushes::load(const std::string& filename)
|
|||||||
|
|
||||||
// Brush image
|
// Brush image
|
||||||
ImageRef image, mask;
|
ImageRef image, mask;
|
||||||
if (TiXmlElement* imageElem = brushElem->FirstChildElement("image"))
|
if (XMLElement* imageElem = brushElem->FirstChildElement("image"))
|
||||||
image = load_xml_image(imageElem);
|
image = load_xml_image(imageElem);
|
||||||
if (TiXmlElement* maskElem = brushElem->FirstChildElement("mask"))
|
if (XMLElement* maskElem = brushElem->FirstChildElement("mask"))
|
||||||
mask = load_xml_image(maskElem);
|
mask = load_xml_image(maskElem);
|
||||||
|
|
||||||
if (image) {
|
if (image) {
|
||||||
@ -351,14 +353,14 @@ void AppBrushes::load(const std::string& filename)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Colors
|
// Colors
|
||||||
if (TiXmlElement* fgcolorElem = brushElem->FirstChildElement("fgcolor")) {
|
if (XMLElement* fgcolorElem = brushElem->FirstChildElement("fgcolor")) {
|
||||||
if (auto value = fgcolorElem->Attribute("value")) {
|
if (auto value = fgcolorElem->Attribute("value")) {
|
||||||
fgColor = app::Color::fromString(value);
|
fgColor = app::Color::fromString(value);
|
||||||
flags |= int(BrushSlot::Flags::FgColor);
|
flags |= int(BrushSlot::Flags::FgColor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TiXmlElement* bgcolorElem = brushElem->FirstChildElement("bgcolor")) {
|
if (XMLElement* bgcolorElem = brushElem->FirstChildElement("bgcolor")) {
|
||||||
if (auto value = bgcolorElem->Attribute("value")) {
|
if (auto value = bgcolorElem->Attribute("value")) {
|
||||||
bgColor = app::Color::fromString(value);
|
bgColor = app::Color::fromString(value);
|
||||||
flags |= int(BrushSlot::Flags::BgColor);
|
flags |= int(BrushSlot::Flags::BgColor);
|
||||||
@ -366,14 +368,14 @@ void AppBrushes::load(const std::string& filename)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Ink
|
// Ink
|
||||||
if (TiXmlElement* inkTypeElem = brushElem->FirstChildElement("inktype")) {
|
if (XMLElement* inkTypeElem = brushElem->FirstChildElement("inktype")) {
|
||||||
if (auto value = inkTypeElem->Attribute("value")) {
|
if (auto value = inkTypeElem->Attribute("value")) {
|
||||||
inkType = app::tools::string_id_to_ink_type(value);
|
inkType = app::tools::string_id_to_ink_type(value);
|
||||||
flags |= int(BrushSlot::Flags::InkType);
|
flags |= int(BrushSlot::Flags::InkType);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TiXmlElement* inkOpacityElem = brushElem->FirstChildElement("inkopacity")) {
|
if (XMLElement* inkOpacityElem = brushElem->FirstChildElement("inkopacity")) {
|
||||||
if (auto value = inkOpacityElem->Attribute("value")) {
|
if (auto value = inkOpacityElem->Attribute("value")) {
|
||||||
inkOpacity = base::convert_to<int>(std::string(value));
|
inkOpacity = base::convert_to<int>(std::string(value));
|
||||||
flags |= int(BrushSlot::Flags::InkOpacity);
|
flags |= int(BrushSlot::Flags::InkOpacity);
|
||||||
@ -381,7 +383,7 @@ void AppBrushes::load(const std::string& filename)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Shade
|
// Shade
|
||||||
if (TiXmlElement* shadeElem = brushElem->FirstChildElement("shade")) {
|
if (XMLElement* shadeElem = brushElem->FirstChildElement("shade")) {
|
||||||
if (auto value = shadeElem->Attribute("value")) {
|
if (auto value = shadeElem->Attribute("value")) {
|
||||||
shade = shade_from_string(value);
|
shade = shade_from_string(value);
|
||||||
flags |= int(BrushSlot::Flags::Shade);
|
flags |= int(BrushSlot::Flags::Shade);
|
||||||
@ -389,7 +391,7 @@ void AppBrushes::load(const std::string& filename)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Pixel-perfect
|
// Pixel-perfect
|
||||||
if (TiXmlElement* pixelPerfectElem = brushElem->FirstChildElement("pixelperfect")) {
|
if (XMLElement* pixelPerfectElem = brushElem->FirstChildElement("pixelperfect")) {
|
||||||
pixelPerfect = bool_attr(pixelPerfectElem, "value", false);
|
pixelPerfect = bool_attr(pixelPerfectElem, "value", false);
|
||||||
flags |= int(BrushSlot::Flags::PixelPerfect);
|
flags |= int(BrushSlot::Flags::PixelPerfect);
|
||||||
}
|
}
|
||||||
@ -414,13 +416,13 @@ void AppBrushes::load(const std::string& filename)
|
|||||||
|
|
||||||
void AppBrushes::save(const std::string& filename) const
|
void AppBrushes::save(const std::string& filename) const
|
||||||
{
|
{
|
||||||
XmlDocumentRef doc(new TiXmlDocument());
|
auto doc = std::make_unique<XMLDocument>();
|
||||||
TiXmlElement brushesElem("brushes");
|
XMLElement* brushesElem = doc->NewElement("brushes");
|
||||||
|
|
||||||
//<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
|
|
||||||
|
doc->InsertEndChild(doc->NewDeclaration("xml version=\"1.0\" encoding=\"utf-8\""));
|
||||||
|
doc->InsertEndChild(brushesElem);
|
||||||
for (const auto& slot : m_slots) {
|
for (const auto& slot : m_slots) {
|
||||||
TiXmlElement brushElem("brush");
|
XMLElement* brushElem = brushesElem->InsertNewChildElement("brush");
|
||||||
if (slot.locked()) {
|
if (slot.locked()) {
|
||||||
// Flags
|
// Flags
|
||||||
int flags = int(slot.flags());
|
int flags = int(slot.flags());
|
||||||
@ -435,32 +437,30 @@ void AppBrushes::save(const std::string& filename) const
|
|||||||
ASSERT(slot.brush());
|
ASSERT(slot.brush());
|
||||||
|
|
||||||
if (flags & int(BrushSlot::Flags::BrushType)) {
|
if (flags & int(BrushSlot::Flags::BrushType)) {
|
||||||
brushElem.SetAttribute(
|
brushElem->SetAttribute(
|
||||||
"type", brush_type_to_string_id(slot.brush()->type()).c_str());
|
"type", brush_type_to_string_id(slot.brush()->type()).c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & int(BrushSlot::Flags::BrushSize)) {
|
if (flags & int(BrushSlot::Flags::BrushSize)) {
|
||||||
brushElem.SetAttribute("size", slot.brush()->size());
|
brushElem->SetAttribute("size", slot.brush()->size());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & int(BrushSlot::Flags::BrushAngle)) {
|
if (flags & int(BrushSlot::Flags::BrushAngle)) {
|
||||||
brushElem.SetAttribute("angle", slot.brush()->angle());
|
brushElem->SetAttribute("angle", slot.brush()->angle());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (slot.brush()->type() == kImageBrushType &&
|
if (slot.brush()->type() == kImageBrushType &&
|
||||||
slot.brush()->originalImage()) {
|
slot.brush()->originalImage()) {
|
||||||
TiXmlElement elem("image");
|
XMLElement* elem = brushElem->InsertNewChildElement("image");
|
||||||
save_xml_image(&elem, slot.brush()->originalImage());
|
save_xml_image(elem, slot.brush()->originalImage());
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
|
|
||||||
if (slot.brush()->maskBitmap()) {
|
if (slot.brush()->maskBitmap()) {
|
||||||
TiXmlElement maskElem("mask");
|
XMLElement* maskElem = brushElem->InsertNewChildElement("mask");
|
||||||
save_xml_image(&maskElem, slot.brush()->maskBitmap());
|
save_xml_image(maskElem, slot.brush()->maskBitmap());
|
||||||
brushElem.InsertEndChild(maskElem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Image color
|
// Image color
|
||||||
brushElem.SetAttribute(
|
brushElem->SetAttribute(
|
||||||
"imagecolor",
|
"imagecolor",
|
||||||
(flags & int(BrushSlot::Flags::ImageColor)) ? "true": "false");
|
(flags & int(BrushSlot::Flags::ImageColor)) ? "true": "false");
|
||||||
}
|
}
|
||||||
@ -468,53 +468,42 @@ void AppBrushes::save(const std::string& filename) const
|
|||||||
|
|
||||||
// Colors
|
// Colors
|
||||||
if (flags & int(BrushSlot::Flags::FgColor)) {
|
if (flags & int(BrushSlot::Flags::FgColor)) {
|
||||||
TiXmlElement elem("fgcolor");
|
XMLElement* elem = brushElem->InsertNewChildElement("fgcolor");
|
||||||
elem.SetAttribute("value", slot.fgColor().toString().c_str());
|
elem->SetAttribute("value", slot.fgColor().toString().c_str());
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & int(BrushSlot::Flags::BgColor)) {
|
if (flags & int(BrushSlot::Flags::BgColor)) {
|
||||||
TiXmlElement elem("bgcolor");
|
XMLElement* elem = brushElem->InsertNewChildElement("bgcolor");
|
||||||
elem.SetAttribute("value", slot.bgColor().toString().c_str());
|
elem->SetAttribute("value", slot.bgColor().toString().c_str());
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ink
|
// Ink
|
||||||
if (flags & int(BrushSlot::Flags::InkType)) {
|
if (flags & int(BrushSlot::Flags::InkType)) {
|
||||||
TiXmlElement elem("inktype");
|
XMLElement* elem = brushElem->InsertNewChildElement("inktype");
|
||||||
elem.SetAttribute(
|
elem->SetAttribute(
|
||||||
"value", app::tools::ink_type_to_string_id(slot.inkType()).c_str());
|
"value", app::tools::ink_type_to_string_id(slot.inkType()).c_str());
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (flags & int(BrushSlot::Flags::InkOpacity)) {
|
if (flags & int(BrushSlot::Flags::InkOpacity)) {
|
||||||
TiXmlElement elem("inkopacity");
|
XMLElement* elem = brushElem->InsertNewChildElement("inkopacity");
|
||||||
elem.SetAttribute("value", slot.inkOpacity());
|
elem->SetAttribute("value", slot.inkOpacity());
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Shade
|
// Shade
|
||||||
if (flags & int(BrushSlot::Flags::Shade)) {
|
if (flags & int(BrushSlot::Flags::Shade)) {
|
||||||
TiXmlElement elem("shade");
|
XMLElement* elem = brushElem->InsertNewChildElement("shade");
|
||||||
elem.SetAttribute("value", shade_to_string(slot.shade()).c_str());
|
elem->SetAttribute("value", shade_to_string(slot.shade()).c_str());
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pixel-perfect
|
// Pixel-perfect
|
||||||
if (flags & int(BrushSlot::Flags::PixelPerfect)) {
|
if (flags & int(BrushSlot::Flags::PixelPerfect)) {
|
||||||
TiXmlElement elem("pixelperfect");
|
XMLElement* elem = brushElem->InsertNewChildElement("pixelperfect");
|
||||||
elem.SetAttribute("value", slot.pixelPerfect() ? "true": "false");
|
elem->SetAttribute("value", slot.pixelPerfect() ? "true": "false");
|
||||||
brushElem.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
brushesElem.InsertEndChild(brushElem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TiXmlDeclaration declaration("1.0", "utf-8", "");
|
save_xml(doc.get(), filename);
|
||||||
doc->InsertEndChild(declaration);
|
|
||||||
doc->InsertEndChild(brushesElem);
|
|
||||||
save_xml(doc, filename);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// static
|
// static
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2019-2023 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -35,7 +35,7 @@
|
|||||||
#include "ui/ui.h"
|
#include "ui/ui.h"
|
||||||
#include "ver/info.h"
|
#include "ver/info.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
@ -47,6 +47,7 @@
|
|||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
|
using namespace tinyxml2;
|
||||||
using namespace ui;
|
using namespace ui;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
@ -343,8 +344,8 @@ void AppMenus::reload()
|
|||||||
{
|
{
|
||||||
MENUS_TRACE("MENUS: AppMenus::reload()");
|
MENUS_TRACE("MENUS: AppMenus::reload()");
|
||||||
|
|
||||||
XmlDocumentRef doc(GuiXml::instance()->doc());
|
XMLDocument* doc = GuiXml::instance()->doc();
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc);
|
||||||
const char* path = GuiXml::instance()->filename();
|
const char* path = GuiXml::instance()->filename();
|
||||||
|
|
||||||
////////////////////////////////////////
|
////////////////////////////////////////
|
||||||
@ -454,9 +455,9 @@ void AppMenus::reload()
|
|||||||
|
|
||||||
LOG("MENU: Loading commands keyboard shortcuts from %s\n", path);
|
LOG("MENU: Loading commands keyboard shortcuts from %s\n", path);
|
||||||
|
|
||||||
TiXmlElement* xmlKey = handle
|
XMLElement* xmlKey = handle
|
||||||
.FirstChild("gui")
|
.FirstChildElement("gui")
|
||||||
.FirstChild("keyboard").ToElement();
|
.FirstChildElement("keyboard").ToElement();
|
||||||
|
|
||||||
// From a fresh start, load the default keys
|
// From a fresh start, load the default keys
|
||||||
KeyboardShortcuts::instance()->clear();
|
KeyboardShortcuts::instance()->clear();
|
||||||
@ -716,15 +717,15 @@ void AppMenus::removeMenuItemFromGroup(Widget* menuItem)
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Menu* AppMenus::loadMenuById(TiXmlHandle& handle, const char* id)
|
Menu* AppMenus::loadMenuById(XMLHandle& handle, const char* id)
|
||||||
{
|
{
|
||||||
ASSERT(id != NULL);
|
ASSERT(id != NULL);
|
||||||
|
|
||||||
// <gui><menus><menu>
|
// <gui><menus><menu>
|
||||||
TiXmlElement* xmlMenu = handle
|
XMLElement* xmlMenu = handle
|
||||||
.FirstChild("gui")
|
.FirstChildElement("gui")
|
||||||
.FirstChild("menus")
|
.FirstChildElement("menus")
|
||||||
.FirstChild("menu").ToElement();
|
.FirstChildElement("menu").ToElement();
|
||||||
while (xmlMenu) {
|
while (xmlMenu) {
|
||||||
const char* menuId = xmlMenu->Attribute("id");
|
const char* menuId = xmlMenu->Attribute("id");
|
||||||
|
|
||||||
@ -739,12 +740,12 @@ Menu* AppMenus::loadMenuById(TiXmlHandle& handle, const char* id)
|
|||||||
throw base::Exception("Error loading menu '%s'\nReinstall the application.", id);
|
throw base::Exception("Error loading menu '%s'\nReinstall the application.", id);
|
||||||
}
|
}
|
||||||
|
|
||||||
Menu* AppMenus::convertXmlelemToMenu(TiXmlElement* elem)
|
Menu* AppMenus::convertXmlelemToMenu(XMLElement* elem)
|
||||||
{
|
{
|
||||||
Menu* menu = new Menu();
|
Menu* menu = new Menu();
|
||||||
menu->setText(m_xmlTranslator(elem, "text"));
|
menu->setText(m_xmlTranslator(elem, "text"));
|
||||||
|
|
||||||
TiXmlElement* child = elem->FirstChildElement();
|
XMLElement* child = elem->FirstChildElement();
|
||||||
while (child) {
|
while (child) {
|
||||||
Widget* menuitem = convertXmlelemToMenuitem(child, menu);
|
Widget* menuitem = convertXmlelemToMenuitem(child, menu);
|
||||||
if (menuitem)
|
if (menuitem)
|
||||||
@ -759,7 +760,7 @@ Menu* AppMenus::convertXmlelemToMenu(TiXmlElement* elem)
|
|||||||
return menu;
|
return menu;
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget* AppMenus::convertXmlelemToMenuitem(TiXmlElement* elem, Menu* menu)
|
Widget* AppMenus::convertXmlelemToMenuitem(XMLElement* elem, Menu* menu)
|
||||||
{
|
{
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
const char* group = elem->Attribute("group");
|
const char* group = elem->Attribute("group");
|
||||||
@ -791,7 +792,7 @@ Widget* AppMenus::convertXmlelemToMenuitem(TiXmlElement* elem, Menu* menu)
|
|||||||
// load params
|
// load params
|
||||||
Params params;
|
Params params;
|
||||||
if (command) {
|
if (command) {
|
||||||
TiXmlElement* xmlParam = elem->FirstChildElement("param");
|
XMLElement* xmlParam = elem->FirstChildElement("param");
|
||||||
while (xmlParam) {
|
while (xmlParam) {
|
||||||
const char* param_name = xmlParam->Attribute("name");
|
const char* param_name = xmlParam->Attribute("name");
|
||||||
const char* param_value = xmlParam->Attribute("value");
|
const char* param_value = xmlParam->Attribute("value");
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2019-2023 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -20,8 +20,10 @@
|
|||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
class TiXmlElement;
|
namespace tinyxml2 {
|
||||||
class TiXmlHandle;
|
class XMLElement;
|
||||||
|
class XMLHandle;
|
||||||
|
}
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
class Command;
|
class Command;
|
||||||
@ -73,9 +75,9 @@ namespace app {
|
|||||||
template<typename Pred>
|
template<typename Pred>
|
||||||
void removeMenuItemFromGroup(Pred pred);
|
void removeMenuItemFromGroup(Pred pred);
|
||||||
|
|
||||||
Menu* loadMenuById(TiXmlHandle& handle, const char *id);
|
Menu* loadMenuById(tinyxml2::XMLHandle& handle, const char *id);
|
||||||
Menu* convertXmlelemToMenu(TiXmlElement* elem);
|
Menu* convertXmlelemToMenu(tinyxml2::XMLElement* elem);
|
||||||
Widget* convertXmlelemToMenuitem(TiXmlElement* elem, Menu* menu);
|
Widget* convertXmlelemToMenuitem(tinyxml2::XMLElement* elem, Menu* menu);
|
||||||
void applyShortcutToMenuitemsWithCommand(Menu* menu, Command* command, const Params& params,
|
void applyShortcutToMenuitemsWithCommand(Menu* menu, Command* command, const Params& params,
|
||||||
const KeyPtr& key);
|
const KeyPtr& key);
|
||||||
void syncNativeMenuItemKeyShortcuts(Menu* menu);
|
void syncNativeMenuItemKeyShortcuts(Menu* menu);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2019-2023 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2017 David Capello
|
// Copyright (C) 2001-2017 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -21,6 +21,8 @@
|
|||||||
#include "fmt/format.h"
|
#include "fmt/format.h"
|
||||||
#include "gfx/color.h"
|
#include "gfx/color.h"
|
||||||
|
|
||||||
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
#include <set>
|
#include <set>
|
||||||
@ -29,6 +31,7 @@ namespace app {
|
|||||||
|
|
||||||
using namespace base;
|
using namespace base;
|
||||||
using namespace doc;
|
using namespace doc;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
@ -89,7 +92,7 @@ template<typename Container,
|
|||||||
typename ChildNameGetterFunc,
|
typename ChildNameGetterFunc,
|
||||||
typename UpdateXmlChildFunc>
|
typename UpdateXmlChildFunc>
|
||||||
void update_xml_collection(const Container& container,
|
void update_xml_collection(const Container& container,
|
||||||
TiXmlElement* xmlParent,
|
XMLElement* xmlParent,
|
||||||
const char* childElemName,
|
const char* childElemName,
|
||||||
const char* idAttrName,
|
const char* idAttrName,
|
||||||
ChildNameGetterFunc childNameGetter,
|
ChildNameGetterFunc childNameGetter,
|
||||||
@ -98,12 +101,12 @@ void update_xml_collection(const Container& container,
|
|||||||
if (!xmlParent)
|
if (!xmlParent)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
TiXmlElement* xmlNext = nullptr;
|
XMLElement* xmlNext = nullptr;
|
||||||
std::set<std::string> existent;
|
std::set<std::string> existent;
|
||||||
|
|
||||||
// Update existent children
|
// Update existent children
|
||||||
for (TiXmlElement* xmlChild=(xmlParent->FirstChild(childElemName) ?
|
for (XMLElement* xmlChild=(xmlParent->FirstChildElement(childElemName) ?
|
||||||
xmlParent->FirstChild(childElemName)->ToElement(): nullptr);
|
xmlParent->FirstChildElement(childElemName): nullptr);
|
||||||
xmlChild;
|
xmlChild;
|
||||||
xmlChild=xmlNext) {
|
xmlChild=xmlNext) {
|
||||||
xmlNext = xmlChild->NextSiblingElement();
|
xmlNext = xmlChild->NextSiblingElement();
|
||||||
@ -126,35 +129,33 @@ void update_xml_collection(const Container& container,
|
|||||||
// Delete this <child> element (as the child was removed from the
|
// Delete this <child> element (as the child was removed from the
|
||||||
// original container)
|
// original container)
|
||||||
if (!found)
|
if (!found)
|
||||||
xmlParent->RemoveChild(xmlChild);
|
xmlParent->DeleteChild(xmlChild);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add new children
|
// Add new children
|
||||||
for (const auto& child : container) {
|
for (const auto& child : container) {
|
||||||
std::string thisChildName = childNameGetter(child);
|
std::string thisChildName = childNameGetter(child);
|
||||||
if (existent.find(thisChildName) == existent.end()) {
|
if (existent.find(thisChildName) == existent.end()) {
|
||||||
TiXmlElement xmlChild(childElemName);
|
XMLElement* xmlChild = xmlParent->InsertNewChildElement(childElemName);
|
||||||
xmlChild.SetAttribute(idAttrName, thisChildName.c_str());
|
xmlChild->SetAttribute(idAttrName, thisChildName.c_str());
|
||||||
updateXmlChild(child, &xmlChild);
|
updateXmlChild(child, xmlChild);
|
||||||
|
|
||||||
xmlParent->InsertEndChild(xmlChild);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void update_xml_part_from_slice_key(const doc::SliceKey* key, TiXmlElement* xmlPart)
|
void update_xml_part_from_slice_key(const doc::SliceKey* key, XMLElement* xmlPart)
|
||||||
{
|
{
|
||||||
xmlPart->SetAttribute("x", key->bounds().x);
|
xmlPart->SetAttribute("x", key->bounds().x);
|
||||||
xmlPart->SetAttribute("y", key->bounds().y);
|
xmlPart->SetAttribute("y", key->bounds().y);
|
||||||
if (!key->hasCenter()) {
|
if (!key->hasCenter()) {
|
||||||
xmlPart->SetAttribute("w", key->bounds().w);
|
xmlPart->SetAttribute("w", key->bounds().w);
|
||||||
xmlPart->SetAttribute("h", key->bounds().h);
|
xmlPart->SetAttribute("h", key->bounds().h);
|
||||||
if (xmlPart->Attribute("w1")) xmlPart->RemoveAttribute("w1");
|
if (xmlPart->Attribute("w1")) xmlPart->DeleteAttribute("w1");
|
||||||
if (xmlPart->Attribute("w2")) xmlPart->RemoveAttribute("w2");
|
if (xmlPart->Attribute("w2")) xmlPart->DeleteAttribute("w2");
|
||||||
if (xmlPart->Attribute("w3")) xmlPart->RemoveAttribute("w3");
|
if (xmlPart->Attribute("w3")) xmlPart->DeleteAttribute("w3");
|
||||||
if (xmlPart->Attribute("h1")) xmlPart->RemoveAttribute("h1");
|
if (xmlPart->Attribute("h1")) xmlPart->DeleteAttribute("h1");
|
||||||
if (xmlPart->Attribute("h2")) xmlPart->RemoveAttribute("h2");
|
if (xmlPart->Attribute("h2")) xmlPart->DeleteAttribute("h2");
|
||||||
if (xmlPart->Attribute("h3")) xmlPart->RemoveAttribute("h3");
|
if (xmlPart->Attribute("h3")) xmlPart->DeleteAttribute("h3");
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
xmlPart->SetAttribute("w1", key->center().x);
|
xmlPart->SetAttribute("w1", key->center().x);
|
||||||
@ -163,8 +164,8 @@ void update_xml_part_from_slice_key(const doc::SliceKey* key, TiXmlElement* xmlP
|
|||||||
xmlPart->SetAttribute("h1", key->center().y);
|
xmlPart->SetAttribute("h1", key->center().y);
|
||||||
xmlPart->SetAttribute("h2", key->center().h);
|
xmlPart->SetAttribute("h2", key->center().h);
|
||||||
xmlPart->SetAttribute("h3", key->bounds().h - key->center().y2());
|
xmlPart->SetAttribute("h3", key->bounds().h - key->center().y2());
|
||||||
if (xmlPart->Attribute("w")) xmlPart->RemoveAttribute("w");
|
if (xmlPart->Attribute("w")) xmlPart->DeleteAttribute("w");
|
||||||
if (xmlPart->Attribute("h")) xmlPart->RemoveAttribute("h");
|
if (xmlPart->Attribute("h")) xmlPart->DeleteAttribute("h");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (key->hasPivot()) {
|
if (key->hasPivot()) {
|
||||||
@ -172,17 +173,17 @@ void update_xml_part_from_slice_key(const doc::SliceKey* key, TiXmlElement* xmlP
|
|||||||
xmlPart->SetAttribute("focusy", key->pivot().y);
|
xmlPart->SetAttribute("focusy", key->pivot().y);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (xmlPart->Attribute("focusx")) xmlPart->RemoveAttribute("focusx");
|
if (xmlPart->Attribute("focusx")) xmlPart->DeleteAttribute("focusx");
|
||||||
if (xmlPart->Attribute("focusy")) xmlPart->RemoveAttribute("focusy");
|
if (xmlPart->Attribute("focusy")) xmlPart->DeleteAttribute("focusy");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void update_xml_slice(const doc::Slice* slice, TiXmlElement* xmlSlice)
|
void update_xml_slice(const doc::Slice* slice, XMLElement* xmlSlice)
|
||||||
{
|
{
|
||||||
if (!slice->userData().text().empty())
|
if (!slice->userData().text().empty())
|
||||||
xmlSlice->SetAttribute("text", slice->userData().text().c_str());
|
xmlSlice->SetAttribute("text", slice->userData().text().c_str());
|
||||||
else if (xmlSlice->Attribute("text"))
|
else if (xmlSlice->Attribute("text"))
|
||||||
xmlSlice->RemoveAttribute("text");
|
xmlSlice->DeleteAttribute("text");
|
||||||
xmlSlice->SetAttribute("color", color_to_hex(slice->userData().color()).c_str());
|
xmlSlice->SetAttribute("color", color_to_hex(slice->userData().color()).c_str());
|
||||||
|
|
||||||
// Update <key> elements
|
// Update <key> elements
|
||||||
@ -192,7 +193,7 @@ void update_xml_slice(const doc::Slice* slice, TiXmlElement* xmlSlice)
|
|||||||
[](const Keyframes<SliceKey>::Key& key) -> std::string {
|
[](const Keyframes<SliceKey>::Key& key) -> std::string {
|
||||||
return base::convert_to<std::string>(key.frame());
|
return base::convert_to<std::string>(key.frame());
|
||||||
},
|
},
|
||||||
[](const Keyframes<SliceKey>::Key& key, TiXmlElement* xmlKey) {
|
[](const Keyframes<SliceKey>::Key& key, XMLElement* xmlKey) {
|
||||||
SliceKey* sliceKey = key.value();
|
SliceKey* sliceKey = key.value();
|
||||||
|
|
||||||
xmlKey->SetAttribute("x", sliceKey->bounds().x);
|
xmlKey->SetAttribute("x", sliceKey->bounds().x);
|
||||||
@ -207,10 +208,10 @@ void update_xml_slice(const doc::Slice* slice, TiXmlElement* xmlSlice)
|
|||||||
xmlKey->SetAttribute("ch", sliceKey->center().h);
|
xmlKey->SetAttribute("ch", sliceKey->center().h);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (xmlKey->Attribute("cx")) xmlKey->RemoveAttribute("cx");
|
if (xmlKey->Attribute("cx")) xmlKey->DeleteAttribute("cx");
|
||||||
if (xmlKey->Attribute("cy")) xmlKey->RemoveAttribute("cy");
|
if (xmlKey->Attribute("cy")) xmlKey->DeleteAttribute("cy");
|
||||||
if (xmlKey->Attribute("cw")) xmlKey->RemoveAttribute("cw");
|
if (xmlKey->Attribute("cw")) xmlKey->DeleteAttribute("cw");
|
||||||
if (xmlKey->Attribute("ch")) xmlKey->RemoveAttribute("ch");
|
if (xmlKey->Attribute("ch")) xmlKey->DeleteAttribute("ch");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sliceKey->hasPivot()) {
|
if (sliceKey->hasPivot()) {
|
||||||
@ -218,8 +219,8 @@ void update_xml_slice(const doc::Slice* slice, TiXmlElement* xmlSlice)
|
|||||||
xmlKey->SetAttribute("py", sliceKey->pivot().y);
|
xmlKey->SetAttribute("py", sliceKey->pivot().y);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (xmlKey->Attribute("px")) xmlKey->RemoveAttribute("px");
|
if (xmlKey->Attribute("px")) xmlKey->DeleteAttribute("px");
|
||||||
if (xmlKey->Attribute("py")) xmlKey->RemoveAttribute("py");
|
if (xmlKey->Attribute("py")) xmlKey->DeleteAttribute("py");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -230,12 +231,12 @@ void load_aseprite_data_file(const std::string& dataFilename,
|
|||||||
doc::Document* doc,
|
doc::Document* doc,
|
||||||
app::Color& defaultSliceColor)
|
app::Color& defaultSliceColor)
|
||||||
{
|
{
|
||||||
XmlDocumentRef xmlDoc = open_xml(dataFilename);
|
XMLDocumentRef xmlDoc = open_xml(dataFilename);
|
||||||
TiXmlHandle handle(xmlDoc.get());
|
XMLHandle handle(xmlDoc.get());
|
||||||
|
|
||||||
TiXmlElement* xmlSlices = handle
|
XMLElement* xmlSlices = handle
|
||||||
.FirstChild("sprite")
|
.FirstChildElement("sprite")
|
||||||
.FirstChild("slices").ToElement();
|
.FirstChildElement("slices").ToElement();
|
||||||
|
|
||||||
// Load slices/parts from theme.xml file
|
// Load slices/parts from theme.xml file
|
||||||
if (xmlSlices &&
|
if (xmlSlices &&
|
||||||
@ -243,13 +244,13 @@ void load_aseprite_data_file(const std::string& dataFilename,
|
|||||||
std::string themeFileName = xmlSlices->Attribute("theme");
|
std::string themeFileName = xmlSlices->Attribute("theme");
|
||||||
|
|
||||||
// Open theme XML file
|
// Open theme XML file
|
||||||
XmlDocumentRef xmlThemeDoc = open_xml(
|
XMLDocumentRef xmlThemeDoc = open_xml(
|
||||||
base::join_path(base::get_file_path(dataFilename), themeFileName));
|
base::join_path(base::get_file_path(dataFilename), themeFileName));
|
||||||
TiXmlHandle themeHandle(xmlThemeDoc.get());
|
XMLHandle themeHandle(xmlThemeDoc.get());
|
||||||
for (TiXmlElement* xmlPart = themeHandle
|
for (XMLElement* xmlPart = themeHandle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("parts")
|
.FirstChildElement("parts")
|
||||||
.FirstChild("part").ToElement();
|
.FirstChildElement("part").ToElement();
|
||||||
xmlPart;
|
xmlPart;
|
||||||
xmlPart=xmlPart->NextSiblingElement()) {
|
xmlPart=xmlPart->NextSiblingElement()) {
|
||||||
const char* partId = xmlPart->Attribute("id");
|
const char* partId = xmlPart->Attribute("id");
|
||||||
@ -300,8 +301,8 @@ void load_aseprite_data_file(const std::string& dataFilename,
|
|||||||
}
|
}
|
||||||
// Load slices from <slice> elements
|
// Load slices from <slice> elements
|
||||||
else if (xmlSlices) {
|
else if (xmlSlices) {
|
||||||
for (TiXmlElement* xmlSlice=(xmlSlices->FirstChild("slice") ?
|
for (XMLElement* xmlSlice=(xmlSlices->FirstChildElement("slice") ?
|
||||||
xmlSlices->FirstChild("slice")->ToElement(): nullptr);
|
xmlSlices->FirstChildElement("slice")->ToElement(): nullptr);
|
||||||
xmlSlice;
|
xmlSlice;
|
||||||
xmlSlice=xmlSlice->NextSiblingElement()) {
|
xmlSlice=xmlSlice->NextSiblingElement()) {
|
||||||
const char* sliceId = xmlSlice->Attribute("id");
|
const char* sliceId = xmlSlice->Attribute("id");
|
||||||
@ -332,8 +333,8 @@ void load_aseprite_data_file(const std::string& dataFilename,
|
|||||||
}
|
}
|
||||||
slice->userData().setColor(color);
|
slice->userData().setColor(color);
|
||||||
|
|
||||||
for (TiXmlElement* xmlKey=(xmlSlice->FirstChild("key") ?
|
for (XMLElement* xmlKey=(xmlSlice->FirstChildElement("key") ?
|
||||||
xmlSlice->FirstChild("key")->ToElement(): nullptr);
|
xmlSlice->FirstChildElement("key")->ToElement(): nullptr);
|
||||||
xmlKey;
|
xmlKey;
|
||||||
xmlKey=xmlKey->NextSiblingElement()) {
|
xmlKey=xmlKey->NextSiblingElement()) {
|
||||||
if (!xmlKey->Attribute("frame"))
|
if (!xmlKey->Attribute("frame"))
|
||||||
@ -373,12 +374,12 @@ void load_aseprite_data_file(const std::string& dataFilename,
|
|||||||
#ifdef ENABLE_SAVE
|
#ifdef ENABLE_SAVE
|
||||||
void save_aseprite_data_file(const std::string& dataFilename, const doc::Document* doc)
|
void save_aseprite_data_file(const std::string& dataFilename, const doc::Document* doc)
|
||||||
{
|
{
|
||||||
XmlDocumentRef xmlDoc = open_xml(dataFilename);
|
XMLDocumentRef xmlDoc = open_xml(dataFilename);
|
||||||
TiXmlHandle handle(xmlDoc.get());
|
XMLHandle handle(xmlDoc.get());
|
||||||
|
|
||||||
TiXmlElement* xmlSlices = handle
|
XMLElement* xmlSlices = handle
|
||||||
.FirstChild("sprite")
|
.FirstChildElement("sprite")
|
||||||
.FirstChild("slices").ToElement();
|
.FirstChildElement("slices").ToElement();
|
||||||
|
|
||||||
// Update theme.xml file
|
// Update theme.xml file
|
||||||
if (xmlSlices &&
|
if (xmlSlices &&
|
||||||
@ -386,13 +387,13 @@ void save_aseprite_data_file(const std::string& dataFilename, const doc::Documen
|
|||||||
// Open theme XML file
|
// Open theme XML file
|
||||||
std::string themeFileName = base::join_path(
|
std::string themeFileName = base::join_path(
|
||||||
base::get_file_path(dataFilename), xmlSlices->Attribute("theme"));
|
base::get_file_path(dataFilename), xmlSlices->Attribute("theme"));
|
||||||
XmlDocumentRef xmlThemeDoc = open_xml(themeFileName);
|
XMLDocumentRef xmlThemeDoc = open_xml(themeFileName);
|
||||||
|
|
||||||
TiXmlHandle themeHandle(xmlThemeDoc.get());
|
XMLHandle themeHandle(xmlThemeDoc.get());
|
||||||
TiXmlElement* xmlParts =
|
XMLElement* xmlParts =
|
||||||
themeHandle
|
themeHandle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("parts").ToElement();
|
.FirstChildElement("parts").ToElement();
|
||||||
|
|
||||||
update_xml_collection(
|
update_xml_collection(
|
||||||
doc->sprite()->slices(),
|
doc->sprite()->slices(),
|
||||||
@ -403,13 +404,13 @@ void save_aseprite_data_file(const std::string& dataFilename, const doc::Documen
|
|||||||
else
|
else
|
||||||
return std::string();
|
return std::string();
|
||||||
},
|
},
|
||||||
[](Slice* slice, TiXmlElement* xmlSlice) {
|
[](Slice* slice, XMLElement* xmlSlice) {
|
||||||
ASSERT(slice->getByFrame(0));
|
ASSERT(slice->getByFrame(0));
|
||||||
update_xml_part_from_slice_key(slice->getByFrame(0), xmlSlice);
|
update_xml_part_from_slice_key(slice->getByFrame(0), xmlSlice);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Save theme.xml file
|
// Save theme.xml file
|
||||||
save_xml(xmlThemeDoc, themeFileName);
|
save_xml(xmlThemeDoc.get(), themeFileName);
|
||||||
}
|
}
|
||||||
// <slices> without "theme" attribute
|
// <slices> without "theme" attribute
|
||||||
else if (xmlSlices) {
|
else if (xmlSlices) {
|
||||||
@ -422,7 +423,7 @@ void save_aseprite_data_file(const std::string& dataFilename, const doc::Documen
|
|||||||
update_xml_slice);
|
update_xml_slice);
|
||||||
|
|
||||||
// Save .aseprite-data file
|
// Save .aseprite-data file
|
||||||
save_xml(xmlDoc, dataFilename);
|
save_xml(xmlDoc.get(), dataFilename);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2020-2022 Igara Studio S.A.
|
// Copyright (C) 2020-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2015 David Capello
|
// Copyright (C) 2001-2015 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -28,8 +28,8 @@ namespace app {
|
|||||||
GuiXml();
|
GuiXml();
|
||||||
|
|
||||||
// Returns the tinyxml document instance.
|
// Returns the tinyxml document instance.
|
||||||
XmlDocumentRef doc() {
|
tinyxml2::XMLDocument* doc() {
|
||||||
return m_doc;
|
return m_doc.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns the name of the gui.xml file.
|
// Returns the name of the gui.xml file.
|
||||||
@ -38,7 +38,7 @@ namespace app {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
XmlDocumentRef m_doc;
|
XMLDocumentRef m_doc;
|
||||||
friend class std::unique_ptr<GuiXml>;
|
friend class std::unique_ptr<GuiXml>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
|
// Copyright (C) 2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2017 David Capello
|
// Copyright (C) 2017 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -11,11 +12,13 @@
|
|||||||
#include "app/i18n/xml_translator.h"
|
#include "app/i18n/xml_translator.h"
|
||||||
|
|
||||||
#include "app/i18n/strings.h"
|
#include "app/i18n/strings.h"
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
std::string XmlTranslator::operator()(const TiXmlElement* elem,
|
using namespace tinyxml2;
|
||||||
|
|
||||||
|
std::string XmlTranslator::operator()(const XMLElement* elem,
|
||||||
const char* attrName)
|
const char* attrName)
|
||||||
{
|
{
|
||||||
const char* value = elem->Attribute(attrName);
|
const char* value = elem->Attribute(attrName);
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
|
// Copyright (C) 2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2017 David Capello
|
// Copyright (C) 2017 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -10,13 +11,15 @@
|
|||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
class TiXmlElement;
|
namespace tinyxml2 {
|
||||||
|
class XMLElement;
|
||||||
|
}
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
class XmlTranslator {
|
class XmlTranslator {
|
||||||
public:
|
public:
|
||||||
std::string operator()(const TiXmlElement* elem,
|
std::string operator()(const tinyxml2::XMLElement* elem,
|
||||||
const char* attrName);
|
const char* attrName);
|
||||||
|
|
||||||
void clearStringIdPrefix();
|
void clearStringIdPrefix();
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2018-2020 Igara Studio S.A.
|
// Copyright (C) 2018-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -29,6 +29,8 @@
|
|||||||
#include "doc/image_impl.h"
|
#include "doc/image_impl.h"
|
||||||
#include "doc/mask.h"
|
#include "doc/mask.h"
|
||||||
|
|
||||||
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
|
|
||||||
@ -41,6 +43,7 @@ namespace app {
|
|||||||
namespace tools {
|
namespace tools {
|
||||||
|
|
||||||
using namespace gfx;
|
using namespace gfx;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
const char* WellKnownTools::RectangularMarquee = "rectangular_marquee";
|
const char* WellKnownTools::RectangularMarquee = "rectangular_marquee";
|
||||||
const char* WellKnownTools::Lasso = "lasso";
|
const char* WellKnownTools::Lasso = "lasso";
|
||||||
@ -206,11 +209,14 @@ void ToolBox::loadTools()
|
|||||||
{
|
{
|
||||||
LOG("TOOL: Loading tools...\n");
|
LOG("TOOL: Loading tools...\n");
|
||||||
|
|
||||||
XmlDocumentRef doc(GuiXml::instance()->doc());
|
XMLDocument* doc = GuiXml::instance()->doc();
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc);
|
||||||
|
|
||||||
// For each group
|
// For each group
|
||||||
TiXmlElement* xmlGroup = handle.FirstChild("gui").FirstChild("tools").FirstChild("group").ToElement();
|
XMLElement* xmlGroup = handle
|
||||||
|
.FirstChildElement("gui")
|
||||||
|
.FirstChildElement("tools")
|
||||||
|
.FirstChildElement("group").ToElement();
|
||||||
while (xmlGroup) {
|
while (xmlGroup) {
|
||||||
const char* groupId = xmlGroup->Attribute("id");
|
const char* groupId = xmlGroup->Attribute("id");
|
||||||
if (!groupId)
|
if (!groupId)
|
||||||
@ -233,8 +239,8 @@ void ToolBox::loadTools()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// For each tool
|
// For each tool
|
||||||
TiXmlNode* xmlToolNode = xmlGroup->FirstChild("tool");
|
XMLNode* xmlToolNode = xmlGroup->FirstChildElement("tool");
|
||||||
TiXmlElement* xmlTool = xmlToolNode ? xmlToolNode->ToElement(): NULL;
|
XMLElement* xmlTool = (xmlToolNode ? xmlToolNode->ToElement(): nullptr);
|
||||||
while (xmlTool) {
|
while (xmlTool) {
|
||||||
const char* toolId = xmlTool->Attribute("id");
|
const char* toolId = xmlTool->Attribute("id");
|
||||||
std::string toolText = m_xmlTranslator(xmlTool, "text");
|
std::string toolText = m_xmlTranslator(xmlTool, "text");
|
||||||
@ -272,7 +278,7 @@ void ToolBox::loadTools()
|
|||||||
LOG("TOOL: Done. %d tools, %d groups.\n", m_tools.size(), m_groups.size());
|
LOG("TOOL: Done. %d tools, %d groups.\n", m_tools.size(), m_groups.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToolBox::loadToolProperties(TiXmlElement* xmlTool, Tool* tool, int button, const std::string& suffix)
|
void ToolBox::loadToolProperties(XMLElement* xmlTool, Tool* tool, int button, const std::string& suffix)
|
||||||
{
|
{
|
||||||
const char* tool_id = tool->getId().c_str();
|
const char* tool_id = tool->getId().c_str();
|
||||||
const char* fill = xmlTool->Attribute(("fill_"+suffix).c_str());
|
const char* fill = xmlTool->Attribute(("fill_"+suffix).c_str());
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2018-2020 Igara Studio S.A.
|
// Copyright (C) 2018-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -16,7 +16,9 @@
|
|||||||
#include "app/i18n/xml_translator.h"
|
#include "app/i18n/xml_translator.h"
|
||||||
#include "app/tools/tool.h"
|
#include "app/tools/tool.h"
|
||||||
|
|
||||||
class TiXmlElement;
|
namespace tinyxml2 {
|
||||||
|
class XMLElement;
|
||||||
|
}
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
namespace tools {
|
namespace tools {
|
||||||
@ -113,7 +115,7 @@ namespace app {
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
void loadTools();
|
void loadTools();
|
||||||
void loadToolProperties(TiXmlElement* xmlTool, Tool* tool, int button, const std::string& suffix);
|
void loadToolProperties(tinyxml2::XMLElement* xmlTool, Tool* tool, int button, const std::string& suffix);
|
||||||
|
|
||||||
std::map<std::string, Ink*> m_inks;
|
std::map<std::string, Ink*> m_inks;
|
||||||
std::map<std::string, Controller*> m_controllers;
|
std::map<std::string, Controller*> m_controllers;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2018-2023 Igara Studio S.A.
|
// Copyright (C) 2018-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -31,6 +31,8 @@
|
|||||||
#include "ui/accelerator.h"
|
#include "ui/accelerator.h"
|
||||||
#include "ui/message.h"
|
#include "ui/message.h"
|
||||||
|
|
||||||
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
@ -39,6 +41,8 @@
|
|||||||
|
|
||||||
#define I18N_KEY(a) app::Strings::keyboard_shortcuts_##a()
|
#define I18N_KEY(a) app::Strings::keyboard_shortcuts_##a()
|
||||||
|
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
struct KeyShortcutAction {
|
struct KeyShortcutAction {
|
||||||
@ -138,7 +142,7 @@ namespace {
|
|||||||
return g_wheel_actions;
|
return g_wheel_actions;
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* get_shortcut(TiXmlElement* elem) {
|
const char* get_shortcut(XMLElement* elem) {
|
||||||
const char* shortcut = NULL;
|
const char* shortcut = NULL;
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
@ -613,13 +617,13 @@ void KeyboardShortcuts::clear()
|
|||||||
m_keys.clear();
|
m_keys.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
void KeyboardShortcuts::importFile(XMLElement* rootElement, KeySource source)
|
||||||
{
|
{
|
||||||
// <keyboard><commands><key>
|
// <keyboard><commands><key>
|
||||||
TiXmlHandle handle(rootElement);
|
XMLHandle handle(rootElement);
|
||||||
TiXmlElement* xmlKey = handle
|
XMLElement* xmlKey = handle
|
||||||
.FirstChild("commands")
|
.FirstChildElement("commands")
|
||||||
.FirstChild("key").ToElement();
|
.FirstChildElement("key").ToElement();
|
||||||
while (xmlKey) {
|
while (xmlKey) {
|
||||||
const char* command_name = xmlKey->Attribute("command");
|
const char* command_name = xmlKey->Attribute("command");
|
||||||
const char* command_key = get_shortcut(xmlKey);
|
const char* command_key = get_shortcut(xmlKey);
|
||||||
@ -637,7 +641,7 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
// Read params
|
// Read params
|
||||||
Params params;
|
Params params;
|
||||||
|
|
||||||
TiXmlElement* xmlParam = xmlKey->FirstChildElement("param");
|
XMLElement* xmlParam = xmlKey->FirstChildElement("param");
|
||||||
while (xmlParam) {
|
while (xmlParam) {
|
||||||
const char* param_name = xmlParam->Attribute("name");
|
const char* param_name = xmlParam->Attribute("name");
|
||||||
const char* param_value = xmlParam->Attribute("value");
|
const char* param_value = xmlParam->Attribute("value");
|
||||||
@ -677,8 +681,8 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
// Load keyboard shortcuts for tools
|
// Load keyboard shortcuts for tools
|
||||||
// <keyboard><tools><key>
|
// <keyboard><tools><key>
|
||||||
xmlKey = handle
|
xmlKey = handle
|
||||||
.FirstChild("tools")
|
.FirstChildElement("tools")
|
||||||
.FirstChild("key").ToElement();
|
.FirstChildElement("key").ToElement();
|
||||||
while (xmlKey) {
|
while (xmlKey) {
|
||||||
const char* tool_id = xmlKey->Attribute("tool");
|
const char* tool_id = xmlKey->Attribute("tool");
|
||||||
const char* tool_key = get_shortcut(xmlKey);
|
const char* tool_key = get_shortcut(xmlKey);
|
||||||
@ -705,8 +709,8 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
// Load keyboard shortcuts for quicktools
|
// Load keyboard shortcuts for quicktools
|
||||||
// <keyboard><quicktools><key>
|
// <keyboard><quicktools><key>
|
||||||
xmlKey = handle
|
xmlKey = handle
|
||||||
.FirstChild("quicktools")
|
.FirstChildElement("quicktools")
|
||||||
.FirstChild("key").ToElement();
|
.FirstChildElement("key").ToElement();
|
||||||
while (xmlKey) {
|
while (xmlKey) {
|
||||||
const char* tool_id = xmlKey->Attribute("tool");
|
const char* tool_id = xmlKey->Attribute("tool");
|
||||||
const char* tool_key = get_shortcut(xmlKey);
|
const char* tool_key = get_shortcut(xmlKey);
|
||||||
@ -733,8 +737,8 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
// Load special keyboard shortcuts for sprite editor customization
|
// Load special keyboard shortcuts for sprite editor customization
|
||||||
// <keyboard><actions><key>
|
// <keyboard><actions><key>
|
||||||
xmlKey = handle
|
xmlKey = handle
|
||||||
.FirstChild("actions")
|
.FirstChildElement("actions")
|
||||||
.FirstChild("key").ToElement();
|
.FirstChildElement("key").ToElement();
|
||||||
while (xmlKey) {
|
while (xmlKey) {
|
||||||
const char* action_id = xmlKey->Attribute("action");
|
const char* action_id = xmlKey->Attribute("action");
|
||||||
const char* action_key = get_shortcut(xmlKey);
|
const char* action_key = get_shortcut(xmlKey);
|
||||||
@ -768,8 +772,8 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
// Load special keyboard shortcuts for mouse wheel customization
|
// Load special keyboard shortcuts for mouse wheel customization
|
||||||
// <keyboard><wheel><key>
|
// <keyboard><wheel><key>
|
||||||
xmlKey = handle
|
xmlKey = handle
|
||||||
.FirstChild("wheel")
|
.FirstChildElement("wheel")
|
||||||
.FirstChild("key").ToElement();
|
.FirstChildElement("key").ToElement();
|
||||||
while (xmlKey) {
|
while (xmlKey) {
|
||||||
const char* action_id = xmlKey->Attribute("action");
|
const char* action_id = xmlKey->Attribute("action");
|
||||||
const char* action_key = get_shortcut(xmlKey);
|
const char* action_key = get_shortcut(xmlKey);
|
||||||
@ -796,8 +800,8 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
// Load special keyboard shortcuts to simulate mouse wheel actions
|
// Load special keyboard shortcuts to simulate mouse wheel actions
|
||||||
// <keyboard><drag><key>
|
// <keyboard><drag><key>
|
||||||
xmlKey = handle
|
xmlKey = handle
|
||||||
.FirstChild("drag")
|
.FirstChildElement("drag")
|
||||||
.FirstChild("key").ToElement();
|
.FirstChildElement("key").ToElement();
|
||||||
while (xmlKey) {
|
while (xmlKey) {
|
||||||
const char* action_id = xmlKey->Attribute("action");
|
const char* action_id = xmlKey->Attribute("action");
|
||||||
const char* action_key = get_shortcut(xmlKey);
|
const char* action_key = get_shortcut(xmlKey);
|
||||||
@ -835,26 +839,25 @@ void KeyboardShortcuts::importFile(TiXmlElement* rootElement, KeySource source)
|
|||||||
|
|
||||||
void KeyboardShortcuts::importFile(const std::string& filename, KeySource source)
|
void KeyboardShortcuts::importFile(const std::string& filename, KeySource source)
|
||||||
{
|
{
|
||||||
XmlDocumentRef doc = app::open_xml(filename);
|
XMLDocumentRef doc = app::open_xml(filename);
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
TiXmlElement* xmlKey = handle.FirstChild("keyboard").ToElement();
|
XMLElement* xmlKey = handle.FirstChildElement("keyboard").ToElement();
|
||||||
|
|
||||||
importFile(xmlKey, source);
|
importFile(xmlKey, source);
|
||||||
}
|
}
|
||||||
|
|
||||||
void KeyboardShortcuts::exportFile(const std::string& filename)
|
void KeyboardShortcuts::exportFile(const std::string& filename)
|
||||||
{
|
{
|
||||||
XmlDocumentRef doc(new TiXmlDocument());
|
auto doc = std::make_unique<XMLDocument>();
|
||||||
|
XMLElement* keyboard = doc->NewElement("keyboard");
|
||||||
|
XMLElement* commands = keyboard->InsertNewChildElement("commands");
|
||||||
|
XMLElement* tools = keyboard->InsertNewChildElement("tools");
|
||||||
|
XMLElement* quicktools = keyboard->InsertNewChildElement("quicktools");
|
||||||
|
XMLElement* actions = keyboard->InsertNewChildElement("actions");
|
||||||
|
XMLElement* wheel = keyboard->InsertNewChildElement("wheel");
|
||||||
|
XMLElement* drag = keyboard->InsertNewChildElement("drag");
|
||||||
|
|
||||||
TiXmlElement keyboard("keyboard");
|
keyboard->SetAttribute("version", XML_KEYBOARD_FILE_VERSION);
|
||||||
TiXmlElement commands("commands");
|
|
||||||
TiXmlElement tools("tools");
|
|
||||||
TiXmlElement quicktools("quicktools");
|
|
||||||
TiXmlElement actions("actions");
|
|
||||||
TiXmlElement wheel("wheel");
|
|
||||||
TiXmlElement drag("drag");
|
|
||||||
|
|
||||||
keyboard.SetAttribute("version", XML_KEYBOARD_FILE_VERSION);
|
|
||||||
|
|
||||||
exportKeys(commands, KeyType::Command);
|
exportKeys(commands, KeyType::Command);
|
||||||
exportKeys(tools, KeyType::Tool);
|
exportKeys(tools, KeyType::Tool);
|
||||||
@ -863,20 +866,12 @@ void KeyboardShortcuts::exportFile(const std::string& filename)
|
|||||||
exportKeys(wheel, KeyType::WheelAction);
|
exportKeys(wheel, KeyType::WheelAction);
|
||||||
exportKeys(drag, KeyType::DragAction);
|
exportKeys(drag, KeyType::DragAction);
|
||||||
|
|
||||||
keyboard.InsertEndChild(commands);
|
doc->InsertEndChild(doc->NewDeclaration("xml version=\"1.0\" encoding=\"utf-8\""));
|
||||||
keyboard.InsertEndChild(tools);
|
|
||||||
keyboard.InsertEndChild(quicktools);
|
|
||||||
keyboard.InsertEndChild(actions);
|
|
||||||
keyboard.InsertEndChild(wheel);
|
|
||||||
keyboard.InsertEndChild(drag);
|
|
||||||
|
|
||||||
TiXmlDeclaration declaration("1.0", "utf-8", "");
|
|
||||||
doc->InsertEndChild(declaration);
|
|
||||||
doc->InsertEndChild(keyboard);
|
doc->InsertEndChild(keyboard);
|
||||||
save_xml(doc, filename);
|
save_xml(doc.get(), filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
void KeyboardShortcuts::exportKeys(TiXmlElement& parent, KeyType type)
|
void KeyboardShortcuts::exportKeys(XMLElement* parent, KeyType type)
|
||||||
{
|
{
|
||||||
for (KeyPtr& key : m_keys) {
|
for (KeyPtr& key : m_keys) {
|
||||||
// Save only user defined accelerators.
|
// Save only user defined accelerators.
|
||||||
@ -893,17 +888,17 @@ void KeyboardShortcuts::exportKeys(TiXmlElement& parent, KeyType type)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void KeyboardShortcuts::exportAccel(TiXmlElement& parent, const Key* key, const ui::Accelerator& accel, bool removed)
|
void KeyboardShortcuts::exportAccel(XMLElement* parent, const Key* key, const ui::Accelerator& accel, bool removed)
|
||||||
{
|
{
|
||||||
TiXmlElement elem("key");
|
XMLElement* elem = parent->InsertNewChildElement("key");
|
||||||
|
|
||||||
switch (key->type()) {
|
switch (key->type()) {
|
||||||
|
|
||||||
case KeyType::Command: {
|
case KeyType::Command: {
|
||||||
elem.SetAttribute("command", key->command()->id().c_str());
|
elem->SetAttribute("command", key->command()->id().c_str());
|
||||||
|
|
||||||
if (key->keycontext() != KeyContext::Any) {
|
if (key->keycontext() != KeyContext::Any) {
|
||||||
elem.SetAttribute("context",
|
elem->SetAttribute("context",
|
||||||
base::convert_to<std::string>(key->keycontext()).c_str());
|
base::convert_to<std::string>(key->keycontext()).c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -911,48 +906,45 @@ void KeyboardShortcuts::exportAccel(TiXmlElement& parent, const Key* key, const
|
|||||||
if (param.second.empty())
|
if (param.second.empty())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
TiXmlElement paramElem("param");
|
XMLElement* paramElem = elem->InsertNewChildElement("param");
|
||||||
paramElem.SetAttribute("name", param.first.c_str());
|
paramElem->SetAttribute("name", param.first.c_str());
|
||||||
paramElem.SetAttribute("value", param.second.c_str());
|
paramElem->SetAttribute("value", param.second.c_str());
|
||||||
elem.InsertEndChild(paramElem);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case KeyType::Tool:
|
case KeyType::Tool:
|
||||||
case KeyType::Quicktool:
|
case KeyType::Quicktool:
|
||||||
elem.SetAttribute("tool", key->tool()->getId().c_str());
|
elem->SetAttribute("tool", key->tool()->getId().c_str());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KeyType::Action:
|
case KeyType::Action:
|
||||||
elem.SetAttribute("action",
|
elem->SetAttribute("action",
|
||||||
base::convert_to<std::string>(key->action()).c_str());
|
base::convert_to<std::string>(key->action()).c_str());
|
||||||
if (key->keycontext() != KeyContext::Any)
|
if (key->keycontext() != KeyContext::Any)
|
||||||
elem.SetAttribute("context",
|
elem->SetAttribute("context",
|
||||||
base::convert_to<std::string>(key->keycontext()).c_str());
|
base::convert_to<std::string>(key->keycontext()).c_str());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KeyType::WheelAction:
|
case KeyType::WheelAction:
|
||||||
elem.SetAttribute("action",
|
elem->SetAttribute("action",
|
||||||
base::convert_to<std::string>(key->wheelAction()));
|
base::convert_to<std::string>(key->wheelAction()).c_str());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KeyType::DragAction:
|
case KeyType::DragAction:
|
||||||
elem.SetAttribute("action",
|
elem->SetAttribute("action",
|
||||||
base::convert_to<std::string>(key->wheelAction()));
|
base::convert_to<std::string>(key->wheelAction()).c_str());
|
||||||
elem.SetAttribute("vector",
|
elem->SetAttribute("vector",
|
||||||
fmt::format("{},{}",
|
fmt::format("{},{}",
|
||||||
key->dragVector().x,
|
key->dragVector().x,
|
||||||
key->dragVector().y));
|
key->dragVector().y).c_str());
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
elem.SetAttribute("shortcut", accel.toString().c_str());
|
elem->SetAttribute("shortcut", accel.toString().c_str());
|
||||||
|
|
||||||
if (removed)
|
if (removed)
|
||||||
elem.SetAttribute("removed", "true");
|
elem->SetAttribute("removed", "true");
|
||||||
|
|
||||||
parent.InsertEndChild(elem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void KeyboardShortcuts::reset()
|
void KeyboardShortcuts::reset()
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2020-2023 Igara Studio S.A.
|
// Copyright (C) 2020-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -12,7 +12,9 @@
|
|||||||
#include "app/ui/key.h"
|
#include "app/ui/key.h"
|
||||||
#include "obs/signal.h"
|
#include "obs/signal.h"
|
||||||
|
|
||||||
class TiXmlElement;
|
namespace tinyxml2 {
|
||||||
|
class XMLElement;
|
||||||
|
}
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
@ -38,7 +40,7 @@ namespace app {
|
|||||||
const bool cloneKeys);
|
const bool cloneKeys);
|
||||||
|
|
||||||
void clear();
|
void clear();
|
||||||
void importFile(TiXmlElement* rootElement, KeySource source);
|
void importFile(tinyxml2::XMLElement* rootElement, KeySource source);
|
||||||
void importFile(const std::string& filename, KeySource source);
|
void importFile(const std::string& filename, KeySource source);
|
||||||
void exportFile(const std::string& filename);
|
void exportFile(const std::string& filename);
|
||||||
void reset();
|
void reset();
|
||||||
@ -78,8 +80,8 @@ namespace app {
|
|||||||
obs::signal<void()> UserChange;
|
obs::signal<void()> UserChange;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void exportKeys(TiXmlElement& parent, KeyType type);
|
void exportKeys(tinyxml2::XMLElement* parent, KeyType type);
|
||||||
void exportAccel(TiXmlElement& parent, const Key* key, const ui::Accelerator& accel, bool removed);
|
void exportAccel(tinyxml2::XMLElement* parent, const Key* key, const ui::Accelerator& accel, bool removed);
|
||||||
|
|
||||||
mutable Keys m_keys;
|
mutable Keys m_keys;
|
||||||
};
|
};
|
||||||
|
@ -27,15 +27,16 @@
|
|||||||
#include "ui/view.h"
|
#include "ui/view.h"
|
||||||
#include "ver/info.h"
|
#include "ver/info.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
using namespace ui;
|
|
||||||
using namespace app::skin;
|
using namespace app::skin;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
using namespace ui;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
@ -267,7 +268,7 @@ void NewsListBox::parseFile(const std::string& filename)
|
|||||||
{
|
{
|
||||||
View* view = View::getView(this);
|
View* view = View::getView(this);
|
||||||
|
|
||||||
XmlDocumentRef doc;
|
XMLDocumentRef doc;
|
||||||
try {
|
try {
|
||||||
doc = open_xml(filename);
|
doc = open_xml(filename);
|
||||||
}
|
}
|
||||||
@ -278,18 +279,18 @@ void NewsListBox::parseFile(const std::string& filename)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
TiXmlElement* itemXml = handle
|
XMLElement* itemXml = handle
|
||||||
.FirstChild("rss")
|
.FirstChildElement("rss")
|
||||||
.FirstChild("channel")
|
.FirstChildElement("channel")
|
||||||
.FirstChild("item").ToElement();
|
.FirstChildElement("item").ToElement();
|
||||||
|
|
||||||
int count = 0;
|
int count = 0;
|
||||||
|
|
||||||
while (itemXml) {
|
while (itemXml) {
|
||||||
TiXmlElement* titleXml = itemXml->FirstChildElement("title");
|
XMLElement* titleXml = itemXml->FirstChildElement("title");
|
||||||
TiXmlElement* descXml = itemXml->FirstChildElement("description");
|
XMLElement* descXml = itemXml->FirstChildElement("description");
|
||||||
TiXmlElement* linkXml = itemXml->FirstChildElement("link");
|
XMLElement* linkXml = itemXml->FirstChildElement("link");
|
||||||
if (titleXml && titleXml->GetText() &&
|
if (titleXml && titleXml->GetText() &&
|
||||||
descXml && descXml->GetText() &&
|
descXml && descXml->GetText() &&
|
||||||
linkXml && linkXml->GetText()) {
|
linkXml && linkXml->GetText()) {
|
||||||
@ -316,10 +317,10 @@ void NewsListBox::parseFile(const std::string& filename)
|
|||||||
itemXml = itemXml->NextSiblingElement();
|
itemXml = itemXml->NextSiblingElement();
|
||||||
}
|
}
|
||||||
|
|
||||||
TiXmlElement* linkXml = handle
|
XMLElement* linkXml = handle
|
||||||
.FirstChild("rss")
|
.FirstChildElement("rss")
|
||||||
.FirstChild("channel")
|
.FirstChildElement("channel")
|
||||||
.FirstChild("link").ToElement();
|
.FirstChildElement("link").ToElement();
|
||||||
if (linkXml && linkXml->GetText())
|
if (linkXml && linkXml->GetText())
|
||||||
addChild(
|
addChild(
|
||||||
new NewsItem(linkXml->GetText(), Strings::news_listbox_more(), ""));
|
new NewsItem(linkXml->GetText(), Strings::news_listbox_more(), ""));
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2019-2023 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -40,7 +40,7 @@
|
|||||||
#include "ui/intern.h"
|
#include "ui/intern.h"
|
||||||
#include "ui/ui.h"
|
#include "ui/ui.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
@ -52,6 +52,7 @@ namespace app {
|
|||||||
namespace skin {
|
namespace skin {
|
||||||
|
|
||||||
using namespace gfx;
|
using namespace gfx;
|
||||||
|
using namespace tinyxml2;
|
||||||
using namespace ui;
|
using namespace ui;
|
||||||
|
|
||||||
// TODO For backward compatibility, in future versions we should remove this (extensions are preferred)
|
// TODO For backward compatibility, in future versions we should remove this (extensions are preferred)
|
||||||
@ -77,7 +78,8 @@ class app::skin::SkinTheme::BackwardCompatibility {
|
|||||||
// Loaded XML <style> element from the original theme (cloned
|
// Loaded XML <style> element from the original theme (cloned
|
||||||
// elements). Must be in order to insert them in the same order in
|
// elements). Must be in order to insert them in the same order in
|
||||||
// the selected theme.
|
// the selected theme.
|
||||||
std::vector<std::unique_ptr<TiXmlElement>> m_styles;
|
XMLDocumentRef m_stylesDoc;
|
||||||
|
std::vector<XMLElement*> m_styles;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void copyingStyles() {
|
void copyingStyles() {
|
||||||
@ -85,13 +87,17 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Called for each <style> element found in theme.xml.
|
// Called for each <style> element found in theme.xml.
|
||||||
void onStyle(TiXmlElement* xmlStyle) {
|
void onStyle(XMLElement* xmlStyle) {
|
||||||
// Loading <style> from the default theme
|
// Loading <style> from the default theme
|
||||||
if (m_state == State::LoadingStyles)
|
if (m_state == State::LoadingStyles) {
|
||||||
m_styles.emplace_back((TiXmlElement*)xmlStyle->Clone());
|
if (!m_stylesDoc)
|
||||||
|
m_stylesDoc = std::make_unique<XMLDocument>();
|
||||||
|
m_styles.emplace_back(
|
||||||
|
xmlStyle->DeepClone(m_stylesDoc.get())->ToElement());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeExistentStyles(TiXmlElement* xmlStyle) {
|
void removeExistentStyles(XMLElement* xmlStyle) {
|
||||||
if (m_state != State::CopyingStyles)
|
if (m_state != State::CopyingStyles)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -117,7 +123,7 @@ public:
|
|||||||
// Copies all missing <style> elements to the new theme. xmlStyles
|
// Copies all missing <style> elements to the new theme. xmlStyles
|
||||||
// is the <styles> element from the theme.xml of the selected theme
|
// is the <styles> element from the theme.xml of the selected theme
|
||||||
// (non the default one).
|
// (non the default one).
|
||||||
void copyMissingStyles(TiXmlNode* xmlStyles) {
|
void copyMissingStyles(XMLNode* xmlStyles) {
|
||||||
if (m_state != State::CopyingStyles)
|
if (m_state != State::CopyingStyles)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -125,8 +131,8 @@ public:
|
|||||||
LOG(VERBOSE, "THEME: Copying <style id='%s'> from default theme\n",
|
LOG(VERBOSE, "THEME: Copying <style id='%s'> from default theme\n",
|
||||||
style->Attribute("id"));
|
style->Attribute("id"));
|
||||||
|
|
||||||
// InsertEndChild() clones the node
|
xmlStyles->InsertEndChild(
|
||||||
xmlStyles->InsertEndChild(*style.get());
|
style->DeepClone(xmlStyles->GetDocument()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -153,7 +159,7 @@ static const char* g_cursor_names[kCursorTypes] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
static FontData* load_font(std::map<std::string, FontData*>& fonts,
|
static FontData* load_font(std::map<std::string, FontData*>& fonts,
|
||||||
const TiXmlElement* xmlFont,
|
const XMLElement* xmlFont,
|
||||||
const std::string& xmlFilename)
|
const std::string& xmlFilename)
|
||||||
{
|
{
|
||||||
const char* fontRef = xmlFont->Attribute("font");
|
const char* fontRef = xmlFont->Attribute("font");
|
||||||
@ -237,8 +243,8 @@ static FontData* load_font(std::map<std::string, FontData*>& fonts,
|
|||||||
font.release();
|
font.release();
|
||||||
|
|
||||||
// Fallback font
|
// Fallback font
|
||||||
const TiXmlElement* xmlFallback =
|
const XMLElement* xmlFallback =
|
||||||
(const TiXmlElement*)xmlFont->FirstChild("fallback");
|
(const XMLElement*)xmlFont->FirstChildElement("fallback");
|
||||||
if (xmlFallback) {
|
if (xmlFallback) {
|
||||||
FontData* fallback = load_font(fonts, xmlFallback, xmlFilename);
|
FontData* fallback = load_font(fonts, xmlFallback, xmlFilename);
|
||||||
if (fallback) {
|
if (fallback) {
|
||||||
@ -345,12 +351,12 @@ void SkinTheme::loadFontData()
|
|||||||
if (!rf.findFirst())
|
if (!rf.findFirst())
|
||||||
throw base::Exception("File %s not found", fontsFilename.c_str());
|
throw base::Exception("File %s not found", fontsFilename.c_str());
|
||||||
|
|
||||||
XmlDocumentRef doc = open_xml(rf.filename());
|
XMLDocumentRef doc = open_xml(rf.filename());
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
|
|
||||||
TiXmlElement* xmlFont = handle
|
XMLElement* xmlFont = handle
|
||||||
.FirstChild("fonts")
|
.FirstChildElement("fonts")
|
||||||
.FirstChild("font").ToElement();
|
.FirstChildElement("font").ToElement();
|
||||||
while (xmlFont) {
|
while (xmlFont) {
|
||||||
load_font(m_fonts, xmlFont, rf.filename());
|
load_font(m_fonts, xmlFont, rf.filename());
|
||||||
xmlFont = xmlFont->NextSiblingElement();
|
xmlFont = xmlFont->NextSiblingElement();
|
||||||
@ -420,15 +426,15 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
// Load the skin XML
|
// Load the skin XML
|
||||||
std::string xml_filename(base::join_path(m_path, "theme.xml"));
|
std::string xml_filename(base::join_path(m_path, "theme.xml"));
|
||||||
|
|
||||||
XmlDocumentRef doc = open_xml(xml_filename);
|
XMLDocumentRef doc = open_xml(xml_filename);
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
|
|
||||||
// Load Preferred scaling
|
// Load Preferred scaling
|
||||||
m_preferredScreenScaling = -1;
|
m_preferredScreenScaling = -1;
|
||||||
m_preferredUIScaling = -1;
|
m_preferredUIScaling = -1;
|
||||||
{
|
{
|
||||||
TiXmlElement* xmlTheme = handle
|
XMLElement* xmlTheme = handle
|
||||||
.FirstChild("theme").ToElement();
|
.FirstChildElement("theme").ToElement();
|
||||||
if (xmlTheme) {
|
if (xmlTheme) {
|
||||||
const char* screenScaling = xmlTheme->Attribute("screenscaling");
|
const char* screenScaling = xmlTheme->Attribute("screenscaling");
|
||||||
const char* uiScaling = xmlTheme->Attribute("uiscaling");
|
const char* uiScaling = xmlTheme->Attribute("uiscaling");
|
||||||
@ -441,10 +447,10 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
|
|
||||||
// Load fonts
|
// Load fonts
|
||||||
{
|
{
|
||||||
TiXmlElement* xmlFont = handle
|
XMLElement* xmlFont = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("fonts")
|
.FirstChildElement("fonts")
|
||||||
.FirstChild("font").ToElement();
|
.FirstChildElement("font").ToElement();
|
||||||
while (xmlFont) {
|
while (xmlFont) {
|
||||||
const char* idStr = xmlFont->Attribute("id");
|
const char* idStr = xmlFont->Attribute("id");
|
||||||
FontData* fontData = load_font(m_fonts, xmlFont, xml_filename);
|
FontData* fontData = load_font(m_fonts, xmlFont, xml_filename);
|
||||||
@ -485,10 +491,10 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
|
|
||||||
// Load dimension
|
// Load dimension
|
||||||
{
|
{
|
||||||
TiXmlElement* xmlDim = handle
|
XMLElement* xmlDim = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("dimensions")
|
.FirstChildElement("dimensions")
|
||||||
.FirstChild("dim").ToElement();
|
.FirstChildElement("dim").ToElement();
|
||||||
while (xmlDim) {
|
while (xmlDim) {
|
||||||
std::string id = xmlDim->Attribute("id");
|
std::string id = xmlDim->Attribute("id");
|
||||||
uint32_t value = strtol(xmlDim->Attribute("value"), NULL, 10);
|
uint32_t value = strtol(xmlDim->Attribute("value"), NULL, 10);
|
||||||
@ -502,10 +508,10 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
|
|
||||||
// Load colors
|
// Load colors
|
||||||
{
|
{
|
||||||
TiXmlElement* xmlColor = handle
|
XMLElement* xmlColor = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("colors")
|
.FirstChildElement("colors")
|
||||||
.FirstChild("color").ToElement();
|
.FirstChildElement("color").ToElement();
|
||||||
while (xmlColor) {
|
while (xmlColor) {
|
||||||
std::string id = xmlColor->Attribute("id");
|
std::string id = xmlColor->Attribute("id");
|
||||||
uint32_t value = strtol(xmlColor->Attribute("value")+1, NULL, 16);
|
uint32_t value = strtol(xmlColor->Attribute("value")+1, NULL, 16);
|
||||||
@ -523,10 +529,10 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
|
|
||||||
// Load parts
|
// Load parts
|
||||||
{
|
{
|
||||||
TiXmlElement* xmlPart = handle
|
XMLElement* xmlPart = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("parts")
|
.FirstChildElement("parts")
|
||||||
.FirstChild("part").ToElement();
|
.FirstChildElement("part").ToElement();
|
||||||
while (xmlPart) {
|
while (xmlPart) {
|
||||||
// Get the tool-icon rectangle
|
// Get the tool-icon rectangle
|
||||||
const char* part_id = xmlPart->Attribute("id");
|
const char* part_id = xmlPart->Attribute("id");
|
||||||
@ -616,10 +622,10 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
|
|
||||||
// Load styles
|
// Load styles
|
||||||
{
|
{
|
||||||
TiXmlElement* xmlStyle = handle
|
XMLElement* xmlStyle = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("styles")
|
.FirstChildElement("styles")
|
||||||
.FirstChild("style").ToElement();
|
.FirstChildElement("style").ToElement();
|
||||||
|
|
||||||
if (!xmlStyle) // Without styles?
|
if (!xmlStyle) // Without styles?
|
||||||
throw base::Exception("There are no styles");
|
throw base::Exception("There are no styles");
|
||||||
@ -752,7 +758,7 @@ void SkinTheme::loadXml(BackwardCompatibility* backward)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TiXmlElement* xmlLayer = xmlStyle->FirstChildElement();
|
XMLElement* xmlLayer = xmlStyle->FirstChildElement();
|
||||||
while (xmlLayer) {
|
while (xmlLayer) {
|
||||||
const std::string layerName = xmlLayer->Value();
|
const std::string layerName = xmlLayer->Value();
|
||||||
|
|
||||||
|
@ -34,7 +34,7 @@
|
|||||||
#include "os/system.h"
|
#include "os/system.h"
|
||||||
#include "ui/ui.h"
|
#include "ui/ui.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
@ -44,11 +44,12 @@
|
|||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
using namespace ui;
|
|
||||||
using namespace app::skin;
|
using namespace app::skin;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
using namespace ui;
|
||||||
|
|
||||||
static int convert_align_value_to_flags(const char *value);
|
static int convert_align_value_to_flags(const char *value);
|
||||||
static int int_attr(const TiXmlElement* elem, const char* attribute_name, int default_value);
|
static int int_attr(const XMLElement* elem, const char* attribute_name, int default_value);
|
||||||
|
|
||||||
WidgetLoader::WidgetLoader()
|
WidgetLoader::WidgetLoader()
|
||||||
: m_tooltipManager(NULL)
|
: m_tooltipManager(NULL)
|
||||||
@ -96,12 +97,12 @@ Widget* WidgetLoader::loadWidgetFromXmlFile(
|
|||||||
m_tooltipManager = NULL;
|
m_tooltipManager = NULL;
|
||||||
m_xmlTranslator.setStringIdPrefix(widgetId.c_str());
|
m_xmlTranslator.setStringIdPrefix(widgetId.c_str());
|
||||||
|
|
||||||
XmlDocumentRef doc(open_xml(xmlFilename));
|
XMLDocumentRef doc = open_xml(xmlFilename);
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
|
|
||||||
// Search the requested widget.
|
// Search the requested widget.
|
||||||
TiXmlElement* xmlElement = handle
|
XMLElement* xmlElement = handle
|
||||||
.FirstChild("gui")
|
.FirstChildElement("gui")
|
||||||
.FirstChildElement().ToElement();
|
.FirstChildElement().ToElement();
|
||||||
|
|
||||||
while (xmlElement) {
|
while (xmlElement) {
|
||||||
@ -118,7 +119,7 @@ Widget* WidgetLoader::loadWidgetFromXmlFile(
|
|||||||
return widget;
|
return widget;
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget* WidgetLoader::convertXmlElementToWidget(const TiXmlElement* elem, Widget* root, Widget* parent, Widget* widget)
|
Widget* WidgetLoader::convertXmlElementToWidget(const XMLElement* elem, Widget* root, Widget* parent, Widget* widget)
|
||||||
{
|
{
|
||||||
const std::string elem_name = elem->Value();
|
const std::string elem_name = elem->Value();
|
||||||
|
|
||||||
@ -542,7 +543,7 @@ Widget* WidgetLoader::convertXmlElementToWidget(const TiXmlElement* elem, Widget
|
|||||||
return widget;
|
return widget;
|
||||||
}
|
}
|
||||||
|
|
||||||
void WidgetLoader::fillWidgetWithXmlElementAttributes(const TiXmlElement* elem, Widget* root, Widget* widget)
|
void WidgetLoader::fillWidgetWithXmlElementAttributes(const XMLElement* elem, Widget* root, Widget* widget)
|
||||||
{
|
{
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
const char* tooltip_dir = elem->Attribute("tooltip_dir");
|
const char* tooltip_dir = elem->Attribute("tooltip_dir");
|
||||||
@ -679,7 +680,7 @@ void WidgetLoader::fillWidgetWithXmlElementAttributes(const TiXmlElement* elem,
|
|||||||
widget->initTheme();
|
widget->initTheme();
|
||||||
}
|
}
|
||||||
|
|
||||||
void WidgetLoader::fillWidgetWithXmlElementAttributesWithChildren(const TiXmlElement* elem, ui::Widget* root, ui::Widget* widget)
|
void WidgetLoader::fillWidgetWithXmlElementAttributesWithChildren(const XMLElement* elem, ui::Widget* root, ui::Widget* widget)
|
||||||
{
|
{
|
||||||
fillWidgetWithXmlElementAttributes(elem, root, widget);
|
fillWidgetWithXmlElementAttributes(elem, root, widget);
|
||||||
|
|
||||||
@ -687,7 +688,7 @@ void WidgetLoader::fillWidgetWithXmlElementAttributesWithChildren(const TiXmlEle
|
|||||||
root = widget;
|
root = widget;
|
||||||
|
|
||||||
// Children
|
// Children
|
||||||
const TiXmlElement* childElem = elem->FirstChildElement();
|
const XMLElement* childElem = elem->FirstChildElement();
|
||||||
while (childElem) {
|
while (childElem) {
|
||||||
Widget* child = convertXmlElementToWidget(childElem, root, widget, NULL);
|
Widget* child = convertXmlElementToWidget(childElem, root, widget, NULL);
|
||||||
if (child) {
|
if (child) {
|
||||||
@ -772,7 +773,7 @@ static int convert_align_value_to_flags(const char *value)
|
|||||||
return flags;
|
return flags;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int int_attr(const TiXmlElement* elem, const char* attribute_name, int default_value)
|
static int int_attr(const XMLElement* elem, const char* attribute_name, int default_value)
|
||||||
{
|
{
|
||||||
const char* value = elem->Attribute(attribute_name);
|
const char* value = elem->Attribute(attribute_name);
|
||||||
|
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
|
// Copyright (C) 2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2017 David Capello
|
// Copyright (C) 2001-2017 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -14,7 +15,9 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
class TiXmlElement;
|
namespace tinyxml2 {
|
||||||
|
class XMLElement;
|
||||||
|
}
|
||||||
|
|
||||||
namespace ui {
|
namespace ui {
|
||||||
class Widget;
|
class Widget;
|
||||||
@ -30,7 +33,7 @@ namespace app {
|
|||||||
public:
|
public:
|
||||||
virtual ~IWidgetTypeCreator() { }
|
virtual ~IWidgetTypeCreator() { }
|
||||||
virtual void dispose() = 0;
|
virtual void dispose() = 0;
|
||||||
virtual ui::Widget* createWidgetFromXml(const TiXmlElement* xmlElem) = 0;
|
virtual ui::Widget* createWidgetFromXml(const tinyxml2::XMLElement* xmlElem) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
WidgetLoader();
|
WidgetLoader();
|
||||||
@ -62,9 +65,9 @@ namespace app {
|
|||||||
const std::string& widgetId,
|
const std::string& widgetId,
|
||||||
ui::Widget* widget);
|
ui::Widget* widget);
|
||||||
|
|
||||||
ui::Widget* convertXmlElementToWidget(const TiXmlElement* elem, ui::Widget* root, ui::Widget* parent, ui::Widget* widget);
|
ui::Widget* convertXmlElementToWidget(const tinyxml2::XMLElement* elem, ui::Widget* root, ui::Widget* parent, ui::Widget* widget);
|
||||||
void fillWidgetWithXmlElementAttributes(const TiXmlElement* elem, ui::Widget* root, ui::Widget* widget);
|
void fillWidgetWithXmlElementAttributes(const tinyxml2::XMLElement* elem, ui::Widget* root, ui::Widget* widget);
|
||||||
void fillWidgetWithXmlElementAttributesWithChildren(const TiXmlElement* elem, ui::Widget* root, ui::Widget* widget);
|
void fillWidgetWithXmlElementAttributesWithChildren(const tinyxml2::XMLElement* elem, ui::Widget* root, ui::Widget* widget);
|
||||||
|
|
||||||
typedef std::map<std::string, IWidgetTypeCreator*> TypeCreatorsMap;
|
typedef std::map<std::string, IWidgetTypeCreator*> TypeCreatorsMap;
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2019 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2015 David Capello
|
// Copyright (C) 2001-2015 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -14,28 +14,28 @@
|
|||||||
#include "app/xml_exception.h"
|
#include "app/xml_exception.h"
|
||||||
#include "base/file_handle.h"
|
#include "base/file_handle.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
using namespace base;
|
using namespace base;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
XmlDocumentRef open_xml(const std::string& filename)
|
XMLDocumentRef open_xml(const std::string& filename)
|
||||||
{
|
{
|
||||||
FileHandle file(open_file(filename, "rb"));
|
FileHandle file(open_file(filename, "rb"));
|
||||||
if (!file)
|
if (!file)
|
||||||
throw Exception("Error loading file: " + filename);
|
throw Exception("Error loading file: " + filename);
|
||||||
|
|
||||||
// Try to load the XML file
|
// Try to load the XML file
|
||||||
auto doc = std::make_shared<TiXmlDocument>();
|
auto doc = std::make_unique<XMLDocument>();
|
||||||
doc->SetValue(filename.c_str());
|
if (doc->LoadFile(file.get()) != XML_SUCCESS)
|
||||||
if (!doc->LoadFile(file.get()))
|
throw XmlException(filename, doc.get());
|
||||||
throw XmlException(doc.get());
|
|
||||||
|
|
||||||
return doc;
|
return doc;
|
||||||
}
|
}
|
||||||
|
|
||||||
void save_xml(XmlDocumentRef doc, const std::string& filename)
|
void save_xml(XMLDocument* doc, const std::string& filename)
|
||||||
{
|
{
|
||||||
FileHandle file(open_file(filename, "wb"));
|
FileHandle file(open_file(filename, "wb"));
|
||||||
if (!file) {
|
if (!file) {
|
||||||
@ -43,11 +43,11 @@ void save_xml(XmlDocumentRef doc, const std::string& filename)
|
|||||||
throw Exception("Error loading file: " + filename);
|
throw Exception("Error loading file: " + filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!doc->SaveFile(file.get()))
|
if (doc->SaveFile(file.get()) != XML_SUCCESS)
|
||||||
throw XmlException(doc.get());
|
throw XmlException(filename, doc);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool bool_attr(const TiXmlElement* elem, const char* attrName, bool defaultVal)
|
bool bool_attr(const XMLElement* elem, const char* attrName, bool defaultVal)
|
||||||
{
|
{
|
||||||
const char* value = elem->Attribute(attrName);
|
const char* value = elem->Attribute(attrName);
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2019 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2015 David Capello
|
// Copyright (C) 2001-2015 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -11,19 +11,21 @@
|
|||||||
|
|
||||||
#include "base/exception.h"
|
#include "base/exception.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
typedef std::shared_ptr<TiXmlDocument> XmlDocumentRef;
|
using XMLDocumentRef = std::unique_ptr<tinyxml2::XMLDocument>;
|
||||||
|
|
||||||
XmlDocumentRef open_xml(const std::string& filename);
|
XMLDocumentRef open_xml(const std::string& filename);
|
||||||
void save_xml(XmlDocumentRef doc, const std::string& filename);
|
void save_xml(tinyxml2::XMLDocument* doc, const std::string& filename);
|
||||||
|
|
||||||
bool bool_attr(const TiXmlElement* elem, const char* attrName, bool defaultVal);
|
bool bool_attr(const tinyxml2::XMLElement* elem,
|
||||||
|
const char* attrName,
|
||||||
|
bool defaultVal);
|
||||||
|
|
||||||
} // namespace app
|
} // namespace app
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2023 Igara Studio S.A.
|
// Copyright (C) 2023-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2015 David Capello
|
// Copyright (C) 2001-2015 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -12,17 +12,21 @@
|
|||||||
#include "app/xml_exception.h"
|
#include "app/xml_exception.h"
|
||||||
|
|
||||||
#include "fmt/format.h"
|
#include "fmt/format.h"
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
XmlException::XmlException(const TiXmlDocument* doc) throw()
|
using namespace tinyxml2;
|
||||||
|
|
||||||
|
XmlException::XmlException(const std::string& filename,
|
||||||
|
const XMLDocument* doc) noexcept
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
setMessage(
|
setMessage(
|
||||||
fmt::format("Error in XML file '{}' (line {}, column {})\nError {}: {}",
|
fmt::format("Error in XML file '{}' (line {})\nError {}: {}",
|
||||||
doc->Value(), doc->ErrorRow(), doc->ErrorCol(),
|
filename, doc->ErrorLineNum(),
|
||||||
doc->ErrorId(), doc->ErrorDesc()).c_str());
|
int(doc->ErrorID()),
|
||||||
|
doc->ErrorStr()).c_str());
|
||||||
}
|
}
|
||||||
catch (...) {
|
catch (...) {
|
||||||
// No throw
|
// No throw
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
|
// Copyright (C) 2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2015 David Capello
|
// Copyright (C) 2001-2015 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -10,13 +11,16 @@
|
|||||||
|
|
||||||
#include "base/exception.h"
|
#include "base/exception.h"
|
||||||
|
|
||||||
class TiXmlDocument;
|
namespace tinyxml2 {
|
||||||
|
class XMLDocument;
|
||||||
|
}
|
||||||
|
|
||||||
namespace app {
|
namespace app {
|
||||||
|
|
||||||
class XmlException : public base::Exception {
|
class XmlException : public base::Exception {
|
||||||
public:
|
public:
|
||||||
XmlException(const TiXmlDocument* doc) throw();
|
XmlException(const std::string& filename,
|
||||||
|
const tinyxml2::XMLDocument* doc) noexcept;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace app
|
} // namespace app
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
// Copyright (c) 2021-2023 Igara Studio S.A.
|
// Copyright (c) 2021-2024 Igara Studio S.A.
|
||||||
// Copyright (c) 2016-2018 David Capello
|
// Copyright (c) 2016-2018 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -15,7 +15,7 @@
|
|||||||
#include "cfg/cfg.h"
|
#include "cfg/cfg.h"
|
||||||
#include "gen/common.h"
|
#include "gen/common.h"
|
||||||
|
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
@ -27,11 +27,12 @@
|
|||||||
// All other translations will be considered work-in-progress.
|
// All other translations will be considered work-in-progress.
|
||||||
#define ENGLISH_ONLY 1
|
#define ENGLISH_ONLY 1
|
||||||
|
|
||||||
typedef std::vector<TiXmlElement*> XmlElements;
|
using namespace tinyxml2;
|
||||||
|
using XmlElements = std::vector<XMLElement*>;
|
||||||
|
|
||||||
static std::string find_first_id(TiXmlElement* elem)
|
static std::string find_first_id(XMLElement* elem)
|
||||||
{
|
{
|
||||||
TiXmlElement* child = elem->FirstChildElement();
|
XMLElement* child = elem->FirstChildElement();
|
||||||
while (child) {
|
while (child) {
|
||||||
const char* id = child->Attribute("id");
|
const char* id = child->Attribute("id");
|
||||||
if (id)
|
if (id)
|
||||||
@ -46,9 +47,9 @@ static std::string find_first_id(TiXmlElement* elem)
|
|||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
static void collect_elements_with_strings(TiXmlElement* elem, XmlElements& elems)
|
static void collect_elements_with_strings(XMLElement* elem, XmlElements& elems)
|
||||||
{
|
{
|
||||||
TiXmlElement* child = elem->FirstChildElement();
|
XMLElement* child = elem->FirstChildElement();
|
||||||
while (child) {
|
while (child) {
|
||||||
const char* text = child->Attribute("text");
|
const char* text = child->Attribute("text");
|
||||||
const char* tooltip = child->Attribute("tooltip");
|
const char* tooltip = child->Attribute("tooltip");
|
||||||
@ -111,19 +112,17 @@ public:
|
|||||||
for (const auto& fn : base::list_files(dir)) {
|
for (const auto& fn : base::list_files(dir)) {
|
||||||
std::string fullFn = base::join_path(dir, fn);
|
std::string fullFn = base::join_path(dir, fn);
|
||||||
base::FileHandle inputFile(base::open_file(fullFn, "rb"));
|
base::FileHandle inputFile(base::open_file(fullFn, "rb"));
|
||||||
std::unique_ptr<TiXmlDocument> doc(new TiXmlDocument());
|
auto doc = std::make_unique<XMLDocument>();
|
||||||
doc->SetValue(fullFn.c_str());
|
if (doc->LoadFile(inputFile.get()) != XML_SUCCESS) {
|
||||||
if (!doc->LoadFile(inputFile.get())) {
|
std::cerr << fullFn << ":"
|
||||||
std::cerr << doc->Value() << ":"
|
<< doc->ErrorLineNum() << ": "
|
||||||
<< doc->ErrorRow() << ":"
|
<< "error " << int(doc->ErrorID()) << ": "
|
||||||
<< doc->ErrorCol() << ": "
|
<< doc->ErrorStr() << "\n";
|
||||||
<< "error " << doc->ErrorId() << ": "
|
|
||||||
<< doc->ErrorDesc() << "\n";
|
|
||||||
|
|
||||||
throw std::runtime_error("invalid input file");
|
throw std::runtime_error("invalid input file");
|
||||||
}
|
}
|
||||||
|
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
XmlElements widgets;
|
XmlElements widgets;
|
||||||
|
|
||||||
const char* warnings = doc->RootElement()->Attribute("i18nwarnings");
|
const char* warnings = doc->RootElement()->Attribute("i18nwarnings");
|
||||||
@ -133,64 +132,63 @@ public:
|
|||||||
m_prefixId = find_first_id(doc->RootElement());
|
m_prefixId = find_first_id(doc->RootElement());
|
||||||
|
|
||||||
collect_elements_with_strings(doc->RootElement(), widgets);
|
collect_elements_with_strings(doc->RootElement(), widgets);
|
||||||
for (TiXmlElement* elem : widgets) {
|
for (XMLElement* elem : widgets) {
|
||||||
checkString(elem, elem->Attribute("text"));
|
checkString(fullFn, elem, elem->Attribute("text"));
|
||||||
checkString(elem, elem->Attribute("tooltip"));
|
checkString(fullFn, elem, elem->Attribute("tooltip"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void checkStringsOnGuiFile(const std::string& fullFn) {
|
void checkStringsOnGuiFile(const std::string& fullFn) {
|
||||||
base::FileHandle inputFile(base::open_file(fullFn, "rb"));
|
base::FileHandle inputFile(base::open_file(fullFn, "rb"));
|
||||||
std::unique_ptr<TiXmlDocument> doc(new TiXmlDocument());
|
auto doc = std::make_unique<XMLDocument>();
|
||||||
doc->SetValue(fullFn.c_str());
|
if (doc->LoadFile(inputFile.get()) != XML_SUCCESS) {
|
||||||
if (!doc->LoadFile(inputFile.get())) {
|
std::cerr << fullFn << ":"
|
||||||
std::cerr << doc->Value() << ":"
|
<< doc->ErrorLineNum() << ": "
|
||||||
<< doc->ErrorRow() << ":"
|
<< "error " << int(doc->ErrorID()) << ": "
|
||||||
<< doc->ErrorCol() << ": "
|
<< doc->ErrorStr() << "\n";
|
||||||
<< "error " << doc->ErrorId() << ": "
|
|
||||||
<< doc->ErrorDesc() << "\n";
|
|
||||||
|
|
||||||
throw std::runtime_error("invalid input file");
|
throw std::runtime_error("invalid input file");
|
||||||
}
|
}
|
||||||
|
|
||||||
TiXmlHandle handle(doc.get());
|
XMLHandle handle(doc.get());
|
||||||
|
|
||||||
// For each menu
|
// For each menu
|
||||||
TiXmlElement* xmlMenu = handle
|
XMLElement* xmlMenu = handle
|
||||||
.FirstChild("gui")
|
.FirstChildElement("gui")
|
||||||
.FirstChild("menus")
|
.FirstChildElement("menus")
|
||||||
.FirstChild("menu").ToElement();
|
.FirstChildElement("menu").ToElement();
|
||||||
while (xmlMenu) {
|
while (xmlMenu) {
|
||||||
const char* menuId = xmlMenu->Attribute("id");
|
const char* menuId = xmlMenu->Attribute("id");
|
||||||
if (menuId) {
|
if (menuId) {
|
||||||
m_prefixId = menuId;
|
m_prefixId = menuId;
|
||||||
XmlElements menus;
|
XmlElements menus;
|
||||||
collect_elements_with_strings(xmlMenu, menus);
|
collect_elements_with_strings(xmlMenu, menus);
|
||||||
for (TiXmlElement* elem : menus)
|
for (XMLElement* elem : menus)
|
||||||
checkString(elem, elem->Attribute("text"));
|
checkString(fullFn, elem, elem->Attribute("text"));
|
||||||
}
|
}
|
||||||
xmlMenu = xmlMenu->NextSiblingElement();
|
xmlMenu = xmlMenu->NextSiblingElement();
|
||||||
}
|
}
|
||||||
|
|
||||||
// For each tool
|
// For each tool
|
||||||
m_prefixId = "tools";
|
m_prefixId = "tools";
|
||||||
TiXmlElement* xmlGroup = handle
|
XMLElement* xmlGroup = handle
|
||||||
.FirstChild("gui")
|
.FirstChildElement("gui")
|
||||||
.FirstChild("tools")
|
.FirstChildElement("tools")
|
||||||
.FirstChild("group").ToElement();
|
.FirstChildElement("group").ToElement();
|
||||||
while (xmlGroup) {
|
while (xmlGroup) {
|
||||||
XmlElements tools;
|
XmlElements tools;
|
||||||
collect_elements_with_strings(xmlGroup, tools);
|
collect_elements_with_strings(xmlGroup, tools);
|
||||||
for (TiXmlElement* elem : tools) {
|
for (XMLElement* elem : tools) {
|
||||||
checkString(elem, elem->Attribute("text"));
|
checkString(fullFn, elem, elem->Attribute("text"));
|
||||||
checkString(elem, elem->Attribute("tooltip"));
|
checkString(fullFn, elem, elem->Attribute("tooltip"));
|
||||||
}
|
}
|
||||||
xmlGroup = xmlGroup->NextSiblingElement();
|
xmlGroup = xmlGroup->NextSiblingElement();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void checkString(TiXmlElement* elem, const char* text) {
|
void checkString(const std::string& filename,
|
||||||
|
XMLElement* elem, const char* text) {
|
||||||
if (!text)
|
if (!text)
|
||||||
return; // Do nothing
|
return; // Do nothing
|
||||||
else if (text[0] == '@') {
|
else if (text[0] == '@') {
|
||||||
@ -212,9 +210,8 @@ public:
|
|||||||
const char* translated =
|
const char* translated =
|
||||||
cfg->getValue(section.c_str(), var.c_str(), nullptr);
|
cfg->getValue(section.c_str(), var.c_str(), nullptr);
|
||||||
if (!translated || translated[0] == 0) {
|
if (!translated || translated[0] == 0) {
|
||||||
std::cerr << elem->GetDocument()->Value() << ":"
|
std::cerr << filename << ":"
|
||||||
<< elem->Row() << ":"
|
<< elem->GetLineNum() << ": "
|
||||||
<< elem->Column() << ": "
|
|
||||||
<< "warning: <" << lang
|
<< "warning: <" << lang
|
||||||
<< "> translation for a string ID wasn't found '"
|
<< "> translation for a string ID wasn't found '"
|
||||||
<< text << "' (" << section << "." << var << ")\n";
|
<< text << "' (" << section << "." << var << ")\n";
|
||||||
@ -224,9 +221,8 @@ public:
|
|||||||
else if (text[0] != '!' &&
|
else if (text[0] != '!' &&
|
||||||
has_alpha_char(text) &&
|
has_alpha_char(text) &&
|
||||||
!is_email(text)) {
|
!is_email(text)) {
|
||||||
std::cerr << elem->GetDocument()->Value() << ":"
|
std::cerr << filename << ":"
|
||||||
<< elem->Row() << ":"
|
<< elem->GetLineNum() << ": "
|
||||||
<< elem->Column() << ": "
|
|
||||||
<< "warning: raw string found '"
|
<< "warning: raw string found '"
|
||||||
<< text << "'\n";
|
<< text << "'\n";
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
// Copyright (c) 2021 Igara Studio S.A.
|
// Copyright (c) 2021-2024 Igara Studio S.A.
|
||||||
// Copyright (c) 2014-2017 David Capello
|
// Copyright (c) 2014-2017 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -15,12 +15,13 @@
|
|||||||
#include "gen/strings_class.h"
|
#include "gen/strings_class.h"
|
||||||
#include "gen/theme_class.h"
|
#include "gen/theme_class.h"
|
||||||
#include "gen/ui_class.h"
|
#include "gen/ui_class.h"
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
typedef base::ProgramOptions PO;
|
using PO = base::ProgramOptions;
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
static void run(int argc, const char* argv[])
|
static void run(int argc, const char* argv[])
|
||||||
{
|
{
|
||||||
@ -38,21 +39,18 @@ static void run(int argc, const char* argv[])
|
|||||||
po.parse(argc, argv);
|
po.parse(argc, argv);
|
||||||
|
|
||||||
// Try to load the XML file
|
// Try to load the XML file
|
||||||
std::unique_ptr<TiXmlDocument> doc;
|
std::unique_ptr<XMLDocument> doc;
|
||||||
|
|
||||||
std::string inputFilename = po.value_of(inputOpt);
|
std::string inputFilename = po.value_of(inputOpt);
|
||||||
if (!inputFilename.empty() &&
|
if (!inputFilename.empty() &&
|
||||||
base::get_file_extension(inputFilename) == "xml") {
|
base::get_file_extension(inputFilename) == "xml") {
|
||||||
base::FileHandle inputFile(base::open_file(inputFilename, "rb"));
|
base::FileHandle inputFile(base::open_file(inputFilename, "rb"));
|
||||||
doc.reset(new TiXmlDocument);
|
doc = std::make_unique<XMLDocument>();
|
||||||
doc->SetValue(inputFilename.c_str());
|
if (doc->LoadFile(inputFile.get()) != XML_SUCCESS) {
|
||||||
if (!doc->LoadFile(inputFile.get())) {
|
std::cerr << inputFilename << ":"
|
||||||
std::cerr << doc->Value() << ":"
|
<< doc->ErrorLineNum() << ": "
|
||||||
<< doc->ErrorRow() << ":"
|
<< "error " << int(doc->ErrorID()) << ": "
|
||||||
<< doc->ErrorCol() << ": "
|
<< doc->ErrorStr() << "\n";
|
||||||
<< "error " << doc->ErrorId() << ": "
|
|
||||||
<< doc->ErrorDesc() << "\n";
|
|
||||||
|
|
||||||
throw std::runtime_error("invalid input file");
|
throw std::runtime_error("invalid input file");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
// Copyright (C) 2019 Igara Studio S.A.
|
// Copyright (C) 2019-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2014-2018 David Capello
|
// Copyright (C) 2014-2018 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -17,9 +17,11 @@
|
|||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
typedef std::vector<TiXmlElement*> XmlElements;
|
using namespace tinyxml2;
|
||||||
|
|
||||||
static void print_pref_class_def(TiXmlElement* elem, const std::string& className, const char* section, int indentSpaces)
|
typedef std::vector<XMLElement*> XmlElements;
|
||||||
|
|
||||||
|
static void print_pref_class_def(XMLElement* elem, const std::string& className, const char* section, int indentSpaces)
|
||||||
{
|
{
|
||||||
std::string indent(indentSpaces, ' ');
|
std::string indent(indentSpaces, ' ');
|
||||||
std::cout
|
std::cout
|
||||||
@ -39,7 +41,7 @@ static void print_pref_class_def(TiXmlElement* elem, const std::string& classNam
|
|||||||
<< indent << " Section* section(const char* id) override;\n"
|
<< indent << " Section* section(const char* id) override;\n"
|
||||||
<< indent << " OptionBase* option(const char* id) override;\n";
|
<< indent << " OptionBase* option(const char* id) override;\n";
|
||||||
|
|
||||||
TiXmlElement* child = (elem->FirstChild() ? elem->FirstChild()->ToElement(): NULL);
|
XMLElement* child = (elem->FirstChild() ? elem->FirstChild()->ToElement(): nullptr);
|
||||||
while (child) {
|
while (child) {
|
||||||
if (child->Value()) {
|
if (child->Value()) {
|
||||||
std::string name = child->Value();
|
std::string name = child->Value();
|
||||||
@ -68,7 +70,7 @@ static void print_pref_class_def(TiXmlElement* elem, const std::string& classNam
|
|||||||
<< indent << "};\n";
|
<< indent << "};\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_pref_class_impl(TiXmlElement* elem, const std::string& prefix, const std::string& className, const char* section)
|
static void print_pref_class_impl(XMLElement* elem, const std::string& prefix, const std::string& className, const char* section)
|
||||||
{
|
{
|
||||||
std::cout
|
std::cout
|
||||||
<< "\n"
|
<< "\n"
|
||||||
@ -79,7 +81,7 @@ static void print_pref_class_impl(TiXmlElement* elem, const std::string& prefix,
|
|||||||
else
|
else
|
||||||
std::cout << " : Section(name)\n";
|
std::cout << " : Section(name)\n";
|
||||||
|
|
||||||
TiXmlElement* child = (elem->FirstChild() ? elem->FirstChild()->ToElement(): NULL);
|
XMLElement* child = (elem->FirstChild() ? elem->FirstChild()->ToElement(): nullptr);
|
||||||
while (child) {
|
while (child) {
|
||||||
if (child->Value()) {
|
if (child->Value()) {
|
||||||
std::string name = child->Value();
|
std::string name = child->Value();
|
||||||
@ -284,7 +286,7 @@ static void print_pref_class_impl(TiXmlElement* elem, const std::string& prefix,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void gen_pref_header(TiXmlDocument* doc, const std::string& inputFn)
|
void gen_pref_header(XMLDocument* doc, const std::string& inputFn)
|
||||||
{
|
{
|
||||||
std::cout
|
std::cout
|
||||||
<< "// Don't modify, generated file from " << inputFn << "\n"
|
<< "// Don't modify, generated file from " << inputFn << "\n"
|
||||||
@ -300,18 +302,18 @@ void gen_pref_header(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
<< "namespace app {\n"
|
<< "namespace app {\n"
|
||||||
<< "namespace gen {\n";
|
<< "namespace gen {\n";
|
||||||
|
|
||||||
TiXmlHandle handle(doc);
|
XMLHandle handle(doc);
|
||||||
TiXmlElement* elem = handle
|
XMLElement* elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("types")
|
.FirstChildElement("types")
|
||||||
.FirstChild("enum").ToElement();
|
.FirstChildElement("enum").ToElement();
|
||||||
while (elem) {
|
while (elem) {
|
||||||
if (!elem->Attribute("id")) throw std::runtime_error("missing 'id' attr in <enum>");
|
if (!elem->Attribute("id")) throw std::runtime_error("missing 'id' attr in <enum>");
|
||||||
std::cout
|
std::cout
|
||||||
<< "\n"
|
<< "\n"
|
||||||
<< " enum class " << elem->Attribute("id") << " {\n";
|
<< " enum class " << elem->Attribute("id") << " {\n";
|
||||||
|
|
||||||
TiXmlElement* child = elem->FirstChildElement("value");
|
XMLElement* child = elem->FirstChildElement("value");
|
||||||
while (child) {
|
while (child) {
|
||||||
if (!child->Attribute("id")) throw std::runtime_error("missing 'id' attr in <value>");
|
if (!child->Attribute("id")) throw std::runtime_error("missing 'id' attr in <value>");
|
||||||
if (!child->Attribute("value")) throw std::runtime_error("missing 'value' attr in <value>");
|
if (!child->Attribute("value")) throw std::runtime_error("missing 'value' attr in <value>");
|
||||||
@ -328,20 +330,20 @@ void gen_pref_header(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
}
|
}
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("global").ToElement();
|
.FirstChildElement("global").ToElement();
|
||||||
if (elem)
|
if (elem)
|
||||||
print_pref_class_def(elem, "GlobalPref", NULL, 2);
|
print_pref_class_def(elem, "GlobalPref", NULL, 2);
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("tool").ToElement();
|
.FirstChildElement("tool").ToElement();
|
||||||
if (elem)
|
if (elem)
|
||||||
print_pref_class_def(elem, "ToolPref", NULL, 2);
|
print_pref_class_def(elem, "ToolPref", NULL, 2);
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("document").ToElement();
|
.FirstChildElement("document").ToElement();
|
||||||
if (elem)
|
if (elem)
|
||||||
print_pref_class_def(elem, "DocPref", NULL, 2);
|
print_pref_class_def(elem, "DocPref", NULL, 2);
|
||||||
|
|
||||||
@ -353,7 +355,7 @@ void gen_pref_header(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
<< "#endif\n";
|
<< "#endif\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
void gen_pref_impl(TiXmlDocument* doc, const std::string& inputFn)
|
void gen_pref_impl(XMLDocument* doc, const std::string& inputFn)
|
||||||
{
|
{
|
||||||
std::cout
|
std::cout
|
||||||
<< "// Don't modify, generated file from " << inputFn << "\n"
|
<< "// Don't modify, generated file from " << inputFn << "\n"
|
||||||
@ -370,22 +372,22 @@ void gen_pref_impl(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
<< "namespace app {\n"
|
<< "namespace app {\n"
|
||||||
<< "namespace gen {\n";
|
<< "namespace gen {\n";
|
||||||
|
|
||||||
TiXmlHandle handle(doc);
|
XMLHandle handle(doc);
|
||||||
TiXmlElement* elem = handle
|
XMLElement* elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("global").ToElement();
|
.FirstChildElement("global").ToElement();
|
||||||
if (elem)
|
if (elem)
|
||||||
print_pref_class_impl(elem, "", "GlobalPref", NULL);
|
print_pref_class_impl(elem, "", "GlobalPref", NULL);
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("tool").ToElement();
|
.FirstChildElement("tool").ToElement();
|
||||||
if (elem)
|
if (elem)
|
||||||
print_pref_class_impl(elem, "", "ToolPref", NULL);
|
print_pref_class_impl(elem, "", "ToolPref", NULL);
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("preferences")
|
.FirstChildElement("preferences")
|
||||||
.FirstChild("document").ToElement();
|
.FirstChildElement("document").ToElement();
|
||||||
if (elem)
|
if (elem)
|
||||||
print_pref_class_impl(elem, "", "DocPref", NULL);
|
print_pref_class_impl(elem, "", "DocPref", NULL);
|
||||||
|
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
|
// Copyright (c) 2024 Igara Studio S.A.
|
||||||
// Copyright (c) 2014 David Capello
|
// Copyright (c) 2014 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -9,9 +10,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
void gen_pref_header(TiXmlDocument* doc, const std::string& inputFn);
|
void gen_pref_header(tinyxml2::XMLDocument* doc, const std::string& inputFn);
|
||||||
void gen_pref_impl(TiXmlDocument* doc, const std::string& inputFn);
|
void gen_pref_impl(tinyxml2::XMLDocument* doc, const std::string& inputFn);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
|
// Copyright (c) 2024 Igara Studio S.A.
|
||||||
// Copyright (c) 2015-2017 David Capello
|
// Copyright (c) 2015-2017 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -13,7 +14,9 @@
|
|||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
void gen_theme_class(TiXmlDocument* doc, const std::string& inputFn)
|
using namespace tinyxml2;
|
||||||
|
|
||||||
|
void gen_theme_class(XMLDocument* doc, const std::string& inputFn)
|
||||||
{
|
{
|
||||||
std::vector<std::string> dimensions;
|
std::vector<std::string> dimensions;
|
||||||
std::vector<std::string> colors;
|
std::vector<std::string> colors;
|
||||||
@ -21,11 +24,11 @@ void gen_theme_class(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
std::vector<std::string> cursors;
|
std::vector<std::string> cursors;
|
||||||
std::vector<std::string> styles;
|
std::vector<std::string> styles;
|
||||||
|
|
||||||
TiXmlHandle handle(doc);
|
XMLHandle handle(doc);
|
||||||
TiXmlElement* elem = handle
|
XMLElement* elem = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("dimensions")
|
.FirstChildElement("dimensions")
|
||||||
.FirstChild("dim").ToElement();
|
.FirstChildElement("dim").ToElement();
|
||||||
while (elem) {
|
while (elem) {
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
dimensions.push_back(id);
|
dimensions.push_back(id);
|
||||||
@ -33,9 +36,9 @@ void gen_theme_class(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
}
|
}
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("colors")
|
.FirstChildElement("colors")
|
||||||
.FirstChild("color").ToElement();
|
.FirstChildElement("color").ToElement();
|
||||||
while (elem) {
|
while (elem) {
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
colors.push_back(id);
|
colors.push_back(id);
|
||||||
@ -43,9 +46,9 @@ void gen_theme_class(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
}
|
}
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("parts")
|
.FirstChildElement("parts")
|
||||||
.FirstChild("part").ToElement();
|
.FirstChildElement("part").ToElement();
|
||||||
while (elem) {
|
while (elem) {
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
if (std::strncmp(id, "cursor_", 7) == 0) {
|
if (std::strncmp(id, "cursor_", 7) == 0) {
|
||||||
@ -57,9 +60,9 @@ void gen_theme_class(TiXmlDocument* doc, const std::string& inputFn)
|
|||||||
}
|
}
|
||||||
|
|
||||||
elem = handle
|
elem = handle
|
||||||
.FirstChild("theme")
|
.FirstChildElement("theme")
|
||||||
.FirstChild("styles")
|
.FirstChildElement("styles")
|
||||||
.FirstChild("style").ToElement();
|
.FirstChildElement("style").ToElement();
|
||||||
while (elem) {
|
while (elem) {
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
styles.push_back(id);
|
styles.push_back(id);
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
|
// Copyright (c) 2024 Igara Studio S.A.
|
||||||
// Copyright (c) 2015-2017 David Capello
|
// Copyright (c) 2015-2017 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -9,8 +10,8 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
void gen_theme_class(TiXmlDocument* doc, const std::string& inputFn);
|
void gen_theme_class(tinyxml2::XMLDocument* doc, const std::string& inputFn);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -17,7 +17,8 @@
|
|||||||
#include <set>
|
#include <set>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
typedef std::vector<TiXmlElement*> XmlElements;
|
using namespace tinyxml2;
|
||||||
|
using XmlElements = std::vector<XMLElement*>;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
@ -36,15 +37,15 @@ struct Item {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static TiXmlElement* find_element_by_id(TiXmlElement* elem, const std::string& thisId)
|
static XMLElement* find_element_by_id(XMLElement* elem, const std::string& thisId)
|
||||||
{
|
{
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
if (id && id == thisId)
|
if (id && id == thisId)
|
||||||
return elem;
|
return elem;
|
||||||
|
|
||||||
TiXmlElement* child = elem->FirstChildElement();
|
XMLElement* child = elem->FirstChildElement();
|
||||||
while (child) {
|
while (child) {
|
||||||
TiXmlElement* match = find_element_by_id(child, thisId);
|
XMLElement* match = find_element_by_id(child, thisId);
|
||||||
if (match)
|
if (match)
|
||||||
return match;
|
return match;
|
||||||
|
|
||||||
@ -54,9 +55,9 @@ static TiXmlElement* find_element_by_id(TiXmlElement* elem, const std::string& t
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void collect_widgets_with_ids(TiXmlElement* elem, XmlElements& widgets)
|
static void collect_widgets_with_ids(XMLElement* elem, XmlElements& widgets)
|
||||||
{
|
{
|
||||||
TiXmlElement* child = elem->FirstChildElement();
|
XMLElement* child = elem->FirstChildElement();
|
||||||
while (child) {
|
while (child) {
|
||||||
const char* id = child->Attribute("id");
|
const char* id = child->Attribute("id");
|
||||||
if (id)
|
if (id)
|
||||||
@ -66,7 +67,7 @@ static void collect_widgets_with_ids(TiXmlElement* elem, XmlElements& widgets)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static Item convert_to_item(TiXmlElement* elem)
|
static Item convert_to_item(XMLElement* elem)
|
||||||
{
|
{
|
||||||
static std::string parent;
|
static std::string parent;
|
||||||
const std::string name = elem->Value();
|
const std::string name = elem->Value();
|
||||||
@ -182,7 +183,7 @@ static Item convert_to_item(TiXmlElement* elem)
|
|||||||
throw base::Exception("Unknown widget name: " + name);
|
throw base::Exception("Unknown widget name: " + name);
|
||||||
}
|
}
|
||||||
|
|
||||||
void gen_ui_class(TiXmlDocument* doc,
|
void gen_ui_class(XMLDocument* doc,
|
||||||
const std::string& inputFn,
|
const std::string& inputFn,
|
||||||
const std::string& widgetId)
|
const std::string& widgetId)
|
||||||
{
|
{
|
||||||
@ -190,8 +191,8 @@ void gen_ui_class(TiXmlDocument* doc,
|
|||||||
<< "// Don't modify, generated file from " << inputFn << "\n"
|
<< "// Don't modify, generated file from " << inputFn << "\n"
|
||||||
<< "\n";
|
<< "\n";
|
||||||
|
|
||||||
TiXmlHandle handle(doc);
|
XMLHandle handle(doc);
|
||||||
TiXmlElement* elem = handle.FirstChild("gui").ToElement();
|
XMLElement* elem = handle.FirstChildElement("gui").ToElement();
|
||||||
elem = find_element_by_id(elem, widgetId);
|
elem = find_element_by_id(elem, widgetId);
|
||||||
if (!elem) {
|
if (!elem) {
|
||||||
std::cout << "#error Widget not found: " << widgetId << "\n";
|
std::cout << "#error Widget not found: " << widgetId << "\n";
|
||||||
@ -202,7 +203,7 @@ void gen_ui_class(TiXmlDocument* doc,
|
|||||||
{
|
{
|
||||||
XmlElements xmlWidgets;
|
XmlElements xmlWidgets;
|
||||||
collect_widgets_with_ids(elem, xmlWidgets);
|
collect_widgets_with_ids(elem, xmlWidgets);
|
||||||
for (TiXmlElement* elem : xmlWidgets) {
|
for (XMLElement* elem : xmlWidgets) {
|
||||||
const char* id = elem->Attribute("id");
|
const char* id = elem->Attribute("id");
|
||||||
if (!id)
|
if (!id)
|
||||||
continue;
|
continue;
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
// Aseprite Code Generator
|
// Aseprite Code Generator
|
||||||
|
// Copyright (c) 2024 Igara Studio S.A.
|
||||||
// Copyright (c) 2014-2016 David Capello
|
// Copyright (c) 2014-2016 David Capello
|
||||||
//
|
//
|
||||||
// This file is released under the terms of the MIT license.
|
// This file is released under the terms of the MIT license.
|
||||||
@ -9,9 +10,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
|
|
||||||
void gen_ui_class(TiXmlDocument* doc,
|
void gen_ui_class(tinyxml2::XMLDocument* doc,
|
||||||
const std::string& inputFn,
|
const std::string& inputFn,
|
||||||
const std::string& widgetId);
|
const std::string& widgetId);
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
// Aseprite
|
// Aseprite
|
||||||
// Copyright (C) 2020 Igara Studio S.A.
|
// Copyright (C) 2020-2024 Igara Studio S.A.
|
||||||
// Copyright (C) 2001-2018 David Capello
|
// Copyright (C) 2001-2018 David Capello
|
||||||
//
|
//
|
||||||
// This program is distributed under the terms of
|
// This program is distributed under the terms of
|
||||||
@ -16,7 +16,7 @@
|
|||||||
#include "net/http_headers.h"
|
#include "net/http_headers.h"
|
||||||
#include "net/http_request.h"
|
#include "net/http_request.h"
|
||||||
#include "net/http_response.h"
|
#include "net/http_response.h"
|
||||||
#include "tinyxml.h"
|
#include "tinyxml2.h"
|
||||||
#include "updater/user_agent.h"
|
#include "updater/user_agent.h"
|
||||||
#include "ver/info.h"
|
#include "ver/info.h"
|
||||||
|
|
||||||
@ -26,6 +26,8 @@
|
|||||||
|
|
||||||
namespace updater {
|
namespace updater {
|
||||||
|
|
||||||
|
using namespace tinyxml2;
|
||||||
|
|
||||||
CheckUpdateResponse::CheckUpdateResponse()
|
CheckUpdateResponse::CheckUpdateResponse()
|
||||||
: m_type(Unknown)
|
: m_type(Unknown)
|
||||||
, m_waitDays(0.0)
|
, m_waitDays(0.0)
|
||||||
@ -44,11 +46,11 @@ CheckUpdateResponse::CheckUpdateResponse(const std::string& responseBody)
|
|||||||
: m_type(Unknown)
|
: m_type(Unknown)
|
||||||
, m_waitDays(0.0)
|
, m_waitDays(0.0)
|
||||||
{
|
{
|
||||||
TiXmlDocument doc;
|
XMLDocument doc;
|
||||||
doc.Parse(responseBody.c_str());
|
doc.Parse(responseBody.c_str());
|
||||||
|
|
||||||
TiXmlHandle handle(&doc);
|
XMLHandle handle(&doc);
|
||||||
TiXmlElement* xmlUpdate = handle.FirstChild("update").ToElement();
|
XMLElement* xmlUpdate = handle.FirstChildElement("update").ToElement();
|
||||||
if (!xmlUpdate) {
|
if (!xmlUpdate) {
|
||||||
// TODO show error?
|
// TODO show error?
|
||||||
return;
|
return;
|
||||||
|
2
third_party/CMakeLists.txt
vendored
2
third_party/CMakeLists.txt
vendored
@ -54,7 +54,7 @@ if(ENABLE_WEBP AND NOT LAF_BACKEND STREQUAL "skia")
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(NOT USE_SHARED_TINYXML)
|
if(NOT USE_SHARED_TINYXML)
|
||||||
add_subdirectory(tinyxml)
|
add_subdirectory(tinyxml2)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(REQUIRE_CURL AND NOT USE_SHARED_CURL)
|
if(REQUIRE_CURL AND NOT USE_SHARED_CURL)
|
||||||
|
12
third_party/tinyxml/CMakeLists.txt
vendored
12
third_party/tinyxml/CMakeLists.txt
vendored
@ -1,12 +0,0 @@
|
|||||||
# ASEPRITE
|
|
||||||
# Copyright (C) 2020 Igara Studio S.A.
|
|
||||||
# Copyright (C) 2001-2013 David Capello
|
|
||||||
|
|
||||||
add_library(tinyxml
|
|
||||||
tinyxml.cpp
|
|
||||||
tinyxmlerror.cpp
|
|
||||||
tinyxmlparser.cpp)
|
|
||||||
|
|
||||||
# Use std::string instead of TiXmlString (we've found some threading
|
|
||||||
# issues related to TiXmlString::nullrep_)
|
|
||||||
target_compile_definitions(tinyxml PUBLIC TIXML_USE_STL)
|
|
298
third_party/tinyxml/changes.txt
vendored
298
third_party/tinyxml/changes.txt
vendored
@ -1,298 +0,0 @@
|
|||||||
Changes in version 1.0.1:
|
|
||||||
- Fixed comment tags which were outputing as '<?--' instead of
|
|
||||||
the correct '<!--'.
|
|
||||||
- Implemented the Next and Prev methods of the TiXmlAttribute class.
|
|
||||||
- Renamed 'LastAttribtute' to 'LastAttribute'
|
|
||||||
- Fixed bad pointer to 'isspace' that could occur while parsing text.
|
|
||||||
- Errors finding beginning and end of tags no longer throw it into an
|
|
||||||
infinite loop. (Hopefully.)
|
|
||||||
|
|
||||||
Changes in version 1.0.2
|
|
||||||
- Minor documentation fixes.
|
|
||||||
|
|
||||||
Changes in version 1.0.3
|
|
||||||
- After nodes are added to a document, they return a pointer
|
|
||||||
to the new node instead of a bool for success.
|
|
||||||
- Elements can be constructed with a value, which is the
|
|
||||||
element name. Every element must have a value or it will be
|
|
||||||
invalid, but the code changes to enforce this are not fully
|
|
||||||
in place.
|
|
||||||
|
|
||||||
Changes in version 1.1.0
|
|
||||||
- Added the TiXmlAttributeSet class to pull the attributes into
|
|
||||||
a seperate container.
|
|
||||||
- Moved the doubly liked list out of XmlBase. Now XmlBase only
|
|
||||||
requires the Print() function and defines some utility functions.
|
|
||||||
- Moved errors into a seperate file. (With the idea of internationalization
|
|
||||||
to the other latin-1 languages.)
|
|
||||||
- Added the "NodeType"
|
|
||||||
- Fixed white space parsing in text to conform with the standard.
|
|
||||||
Basically, all white space becomes just one space.
|
|
||||||
- Added the TiXmlDeclaration class to read xml declarations.
|
|
||||||
|
|
||||||
Changes in version 1.2.0
|
|
||||||
- Removed the factory. The factory was not really in the spirit
|
|
||||||
of small and simple, confused the code, and was of limited value.
|
|
||||||
- Added FirstChildElement and NextSiblingElement, because they
|
|
||||||
are such common functions.
|
|
||||||
- Re-wrote the example to test and demonstrate more functionality.
|
|
||||||
|
|
||||||
Changes in version 1.2.1
|
|
||||||
- Fixed a bug where comments couldn't be inside elements.
|
|
||||||
- Loading now clears out existing XML rather than appending.
|
|
||||||
- Added the "Clear" method on a node to delete all its children.
|
|
||||||
|
|
||||||
Changes in version 1.2.2
|
|
||||||
- Fixed TiXmlAttribute::Previous actually returning "next." Thanks
|
|
||||||
to Rickard Troedsson for the bug fix.
|
|
||||||
|
|
||||||
Changes in version 1.2.3
|
|
||||||
- Added the TIXML prefix to the error strings to resolve conflicts
|
|
||||||
with #defines in OS headers. Thanks to Steve Lhomme.
|
|
||||||
- Fixed a delete buf that should be a delete [] buf.
|
|
||||||
Thanks to Ephi Sinowitz.
|
|
||||||
|
|
||||||
Changes in version 1.2.4
|
|
||||||
- ReplaceChild() was almost guarenteed to fail. Should be fixed,
|
|
||||||
thanks to Joe Smith. Joe also pointed out that the Print() functions
|
|
||||||
should take stream references: I agree, and would like to overload
|
|
||||||
the Print() method to take either format, but I don't want to do
|
|
||||||
this in a dot release.
|
|
||||||
- Some compilers seem to need an extra <ctype.h> include. Thanks
|
|
||||||
to Steve Lhomme for that.
|
|
||||||
|
|
||||||
Changes in version 2.0.0 BETA
|
|
||||||
- Made the ToXXX() casts safe if 'this' is null.
|
|
||||||
When "LoadFile" is called with a filename, the value will correctly get set.
|
|
||||||
Thanks to Brian Yoder.
|
|
||||||
- Fixed bug where isalpha() and isalnum() would get called with a negative value for
|
|
||||||
high ascii numbers. Thanks to Alesky Aksenov.
|
|
||||||
- Fixed some errors codes that were not getting set.
|
|
||||||
- Made methods "const" that were not.
|
|
||||||
- Added a switch to enable or disable the ignoring of white space. ( TiXmlDocument::SetIgnoreWhiteSpace() )
|
|
||||||
- Greater standardization and code re-use in the parser.
|
|
||||||
- Added a stream out operator.
|
|
||||||
- Added a stream in operator.
|
|
||||||
- Entity support, of predefined entites. &#x entities are untouched by input or output.
|
|
||||||
- Improved text out formatting.
|
|
||||||
- Fixed ReplaceChild bug, thanks to Tao Chen.
|
|
||||||
|
|
||||||
Changes in version 2.0.1
|
|
||||||
- Fixed hanging on loading a 0 length file. Thanks to Jeff Scozzafava.
|
|
||||||
- Fixed crashing on InsertBeforeChild and InsertAfterChild. Also possibility of bad links being
|
|
||||||
created by same function. Thanks to Frank De prins.
|
|
||||||
- Added missing licence text. Thanks to Lars Willemsens.
|
|
||||||
- Added <ctype.h> include, at the suggestion of Steve Walters.
|
|
||||||
|
|
||||||
Changes in version 2.1.0
|
|
||||||
- Yves Berquin brings us the STL switch. The forum on SourceForge, and various emails to
|
|
||||||
me, have long debated all out STL vs. no STL at all. And now you can have it both ways.
|
|
||||||
TinyXml will compile either way.
|
|
||||||
|
|
||||||
Changes in version 2.1.1
|
|
||||||
- Compilation warnings.
|
|
||||||
|
|
||||||
Changes in version 2.1.2
|
|
||||||
- Uneeded code is not compiled in the STL case.
|
|
||||||
- Changed headers so that STL can be turned on or off in tinyxml.h
|
|
||||||
|
|
||||||
Changes in version 2.1.3
|
|
||||||
- Fixed non-const reference in API; now uses a pointer.
|
|
||||||
- Copy constructor of TiXmlString not checking for assignment to self.
|
|
||||||
- Nimrod Cohen found a truly evil bug in the STL implementation that occurs
|
|
||||||
when a string is converted to a c_str and then assigned to self. Search for
|
|
||||||
STL_STRING_BUG for a full description. I'm asserting this is a Microsoft STL
|
|
||||||
bug, since &string and string.c_str() should never be the same. Nevertheless,
|
|
||||||
the code works around it.
|
|
||||||
- Urivan Saaib pointed out a compiler conflict, where the C headers define
|
|
||||||
the isblank macro, which was wiping out the TiXmlString::isblank() method.
|
|
||||||
The method was unused and has been removed.
|
|
||||||
|
|
||||||
Changes in version 2.1.4
|
|
||||||
- Reworked the entity code. Entities were not correctly surving round trip input and output.
|
|
||||||
Will now automatically create entities for high ascii in output.
|
|
||||||
|
|
||||||
Changes in version 2.1.5
|
|
||||||
- Bug fix by kylotan : infinite loop on some input (tinyxmlparser.cpp rev 1.27)
|
|
||||||
- Contributed by Ivica Aracic (bytelord) : 1 new VC++ project to compile versions as static libraries (tinyxml_lib.dsp),
|
|
||||||
and an example usage in xmltest.dsp
|
|
||||||
(Patch request ID 678605)
|
|
||||||
- A suggestion by Ronald Fenner Jr (dormlock) to add #include <istream> and <ostream> for Apple's Project Builder
|
|
||||||
(Patch request ID 697642)
|
|
||||||
- A patch from ohommes that allows to parse correctly dots in element names and attribute names
|
|
||||||
(Patch request 602600 and kylotan 701728)
|
|
||||||
- A patch from hermitgeek ( James ) and wasteland for improper error reporting
|
|
||||||
- Reviewed by Lee, with the following changes:
|
|
||||||
- Got sick of fighting the STL/non-STL thing in the windows build. Broke
|
|
||||||
them out as seperate projects.
|
|
||||||
- I have too long not included the dsw. Added.
|
|
||||||
- TinyXmlText had a protected Print. Odd.
|
|
||||||
- Made LinkEndChild public, with docs and appropriate warnings.
|
|
||||||
- Updated the docs.
|
|
||||||
|
|
||||||
2.2.0
|
|
||||||
- Fixed an uninitialized pointer in the TiXmlAttributes
|
|
||||||
- Fixed STL compilation problem in MinGW (and gcc 3?) - thanks Brian Yoder for finding this one
|
|
||||||
- Fixed a syntax error in TiXmlDeclaration - thanks Brian Yoder
|
|
||||||
- Fletcher Dunn proposed and submitted new error handling that tracked the row and column. Lee
|
|
||||||
modified it to not have performance impact.
|
|
||||||
- General cleanup suggestions from Fletcher Dunn.
|
|
||||||
- In error handling, general errors will no longer clear the error state of specific ones.
|
|
||||||
- Fix error in documentation : comments starting with "<?--" instead of "<!--" (thanks ion_pulse)
|
|
||||||
- Added the TiXmlHandle. An easy, safe way to browse XML DOMs with less code.
|
|
||||||
- Added QueryAttribute calls which have better error messaging. (Proposed by Fletcher Dunn)
|
|
||||||
- Nodes and attributes can now print themselves to strings. (Yves suggestion)
|
|
||||||
- Fixed bug where entities with one character would confuse parser. (Thanks Roman)
|
|
||||||
|
|
||||||
2.2.1
|
|
||||||
- Additional testing (no more bugs found to be fixed in this release)
|
|
||||||
- Significant performance improvement to the cursor code.
|
|
||||||
|
|
||||||
2.3.0
|
|
||||||
- User Data are now defined in TiXmlBase instead of TiXmlNode
|
|
||||||
- Character Entities are now UCS-2
|
|
||||||
- Character Entities can be decimal or hexadecimal
|
|
||||||
- UTF-8 conversion.
|
|
||||||
- Fixed many, many bugs.
|
|
||||||
|
|
||||||
2.3.1
|
|
||||||
- Fixed bug in handling nulls embedded in the input.
|
|
||||||
- Make UTF-8 parser tolerant of bad text encoding.
|
|
||||||
- Added encoding detection.
|
|
||||||
- Many fixes and input from John-Philip Leonard Johansson (JP) and Ellers,
|
|
||||||
including UTF-8 feedback, bug reports, and patches. Thanks!
|
|
||||||
- Added version # constants - a suggestion from JP and Ellers.
|
|
||||||
- [ 979180 ] Missing ; in entity reference, fix from Rob Laveaux.
|
|
||||||
- Copy constructors and assignment have been a long time coming. Thanks to
|
|
||||||
Fokke and JP.
|
|
||||||
|
|
||||||
2.3.2
|
|
||||||
- Made the IsAlpha and IsAlphaNum much more tolerant of non-UTF-8 encodings. Thanks
|
|
||||||
Volker Boerchers for finding the issue.
|
|
||||||
- Ran the program though the magnificent Valgrind - http://valgrind.kde.org - to check
|
|
||||||
for memory errors. Fixed some minor issues.
|
|
||||||
|
|
||||||
2.3.3
|
|
||||||
- Fixed crash when test program was run from incorrect directory.
|
|
||||||
- Fixed bug 1070717 - empty document not returned correctly - thanks Katsuhisa Yuasa.
|
|
||||||
- Bug 1079301 resolved - deprecated stdlib calls. Thanks Adrian Boeing.
|
|
||||||
- Bug 1035218 fixed - documentation errors. Xunji Luo
|
|
||||||
- Other bug fixes have accumulated and been fixed on the way as well; my apologies to
|
|
||||||
authors not credited!
|
|
||||||
- Big fix / addition is to correctly return const values. TinyXml could basically
|
|
||||||
remove const in a method like this: TiXmlElement* Foo() const, where the returned element
|
|
||||||
was a pointer to internal data. That is now: const TiXmlElement* Foo() const and
|
|
||||||
TiXmlElement* Foo().
|
|
||||||
|
|
||||||
2.3.4
|
|
||||||
- Fixed additional const errors, thanks Kent Gibson.
|
|
||||||
- Correctly re-enable warnings after tinyxml header. Thanks Cory Nelson.
|
|
||||||
- Variety of type cleanup and warning fixes. Thanks Warren Stevens.
|
|
||||||
- Cleaned up unneeded constructor calls in TinyString - thanks to Geoff Carlton and
|
|
||||||
the discussion group on sourceforge.
|
|
||||||
|
|
||||||
2.4.0
|
|
||||||
- Improved string class, thanks Tyge Lovset (whose name gets mangled in English - sorry)
|
|
||||||
- Type cast compiler warning, thanks Rob van den Bogaard
|
|
||||||
- Added GetText() convenience function. Thanks Ilya Parniuk & Andrew Ellers for input.
|
|
||||||
- Many thanks to marlonism for finding an infinite loop in bad xml.
|
|
||||||
- A patch to cleanup warnings from Robert Gebis.
|
|
||||||
- Added ValueStr() to get the value of a node as a string.
|
|
||||||
- TiXmlText can now parse and output as CDATA
|
|
||||||
- Additional string improvement from James (z2895)
|
|
||||||
- Removed extraneous 'const', thanks David Aldrich
|
|
||||||
- First pass at switching to the "safe" stdlib functions. Many people have suggested and
|
|
||||||
pushed on this, but Warren Stevens put together the first proposal.
|
|
||||||
- TinyXml now will do EOL normalization before parsing, consistent with the W3C XML spec.
|
|
||||||
- Documents loaded with the UTF-8 BOM will now save with the UTF-8 BOM. Good suggestion
|
|
||||||
from 'instructor_'
|
|
||||||
- Ellers submitted his very popular tutorials, which have been added to the distribution.
|
|
||||||
|
|
||||||
2.4.1
|
|
||||||
- Fixed CDATA output formatting
|
|
||||||
- Fixed memory allocators in TinyString to work with overloaded new/delete
|
|
||||||
|
|
||||||
2.4.2
|
|
||||||
- solosnake pointed out that TIXML_LOG causes problems on an XBOX. The definition in the header
|
|
||||||
was superflous and was moved inside of DEBUG_PARSING
|
|
||||||
|
|
||||||
2.4.3
|
|
||||||
- Fixed a test bug that caused a crash in 'xmltest'. TinyXML was fine, but it isn't good
|
|
||||||
to ship with a broken test suite.
|
|
||||||
- Started converting some functions to not cast between std::string and const char*
|
|
||||||
quite as often.
|
|
||||||
- Added FILE* versions of the document loads - good suggestion from Wade Brainerd
|
|
||||||
- Empty documents might not always return the errors they should. [1398915] Thanks to igor v.
|
|
||||||
- Added some asserts for multiply adding a node, regardng bug [1391937] suggested by Paco Arjonilla.
|
|
||||||
|
|
||||||
2.4.4
|
|
||||||
- Bug find thanks to andre-gross found a memory leak that occured when a document failed to load.
|
|
||||||
- Bug find (and good analysis) by VirtualJim who found a case where attribute parsing
|
|
||||||
should be throwing an error and wasn't.
|
|
||||||
- Steve Hyatt suggested the QueryValueAttribute method, which is now implemented.
|
|
||||||
- DavidA identified a chunk of dead code.
|
|
||||||
- Andrew Baxter sent in some compiler warnings that were good clean up points.
|
|
||||||
|
|
||||||
2.5
|
|
||||||
- Added the Visit() API. Many thanks to both Andrew Ellerton and John-Philip for all their
|
|
||||||
work, code, suggestion, and just general pushing that it should be done.
|
|
||||||
- Removed existing streaming code and use TiXmlPrinter instead.
|
|
||||||
- [ tinyxml-Bugs-1527079 ] Compile error in tinystr.cpp fixed, thanks to Paul Suggs
|
|
||||||
- [ tinyxml-Bugs-1522890 ] SaveFile has no error checks fixed, thanks to Ivan Dobrokotov
|
|
||||||
- Ivan Dobrokotov also reported redundant memory allocation in the Attribute() method, which
|
|
||||||
upon investigation was a mess. The attribute should now be fixed for both const char* and
|
|
||||||
std::string, and the return types match the input parameters.
|
|
||||||
- Feature [ 1511105 ] Make TiXmlComment constructor accept a string / char*, implemented.
|
|
||||||
Thanks to Karl Itschen for the feedback.
|
|
||||||
- [ 1480108 ] Stream parsing fails when CDATA contains tags was found by Tobias Grimm, who also
|
|
||||||
submitted a test case and patch. A significant bug in CDATA streaming (operator>>) has now
|
|
||||||
been fixed.
|
|
||||||
|
|
||||||
2.5.2
|
|
||||||
- Lieven, and others, pointed out a missing const-cast that upset the Open Watcom compiler.
|
|
||||||
Should now be fixed.
|
|
||||||
- ErrorRow and ErrorCol should have been const, and weren't. Fixed thanks to Dmitry Polutov.
|
|
||||||
|
|
||||||
2.5.3
|
|
||||||
- zloe_zlo identified a missing string specialization for QueryValueAttribute() [ 1695429 ]. Worked
|
|
||||||
on this bug, but not sure how to fix it in a safe, cross-compiler way.
|
|
||||||
- increased warning level to 4 and turned on detect 64 bit portability issues for VC2005.
|
|
||||||
May address [ 1677737 ] VS2005: /Wp64 warnings
|
|
||||||
- grosheck identified several problems with the Document copy. Many thanks for [ 1660367 ]
|
|
||||||
- Nice catch, and suggested fix, be Gilad Novik on the Printer dropping entities.
|
|
||||||
"[ 1600650 ] Bug when printing xml text" is now fixed.
|
|
||||||
- A subtle fix from Nicos Gollan in the tinystring initializer:
|
|
||||||
[ 1581449 ] Fix initialiser of TiXmlString::nullrep_
|
|
||||||
- Great catch, although there isn't a submitter for the bug. [ 1475201 ] TinyXML parses entities in comments.
|
|
||||||
Comments should not, in fact, parse entities. Fixed the code path and added tests.
|
|
||||||
- We were not catching all the returns from ftell. Thanks to Bernard for catching that.
|
|
||||||
|
|
||||||
2.5.4
|
|
||||||
- A TiXMLDocument can't be a sub-node. Block this from happening in the 'replace'. Thanks Noam.
|
|
||||||
- [ 1714831 ] TiXmlBase::location is not copied by copy-ctors, fix reported and suggested by Nicola Civran.
|
|
||||||
- Fixed possible memory overrun in the comment reading code - thanks gcarlton77
|
|
||||||
|
|
||||||
2.5.5
|
|
||||||
- Alex van der Wal spotted incorrect types (lf) being used in print and scan. robertnestor pointed out some problems with the simple solution. Types updated.
|
|
||||||
- Johannes Hillert pointed out some bug typos.
|
|
||||||
- Christian Mueller identified inconsistent error handling with Attributes.
|
|
||||||
- olivier barthelemy also reported a problem with double truncation, also related to the %lf issue.
|
|
||||||
- zaelsius came up with a great (and simple) suggestion to fix QueryValueAttribute truncating strings.
|
|
||||||
- added some null pointer checks suggested by hansenk
|
|
||||||
- Sami Väisänen found a (rare) buffer overrun that could occur in parsing.
|
|
||||||
- vi tri filed a bug that led to a refactoring of the attribute setting mess (as well as adding a missing SetDoubleAttribute() )
|
|
||||||
- removed TIXML_ERROR_OUT_OF_MEMORY. TinyXML does not systematically address OOO, and the notion it does is misleading.
|
|
||||||
- vanneto, keithmarshall, others all reported the warning from IsWhiteSpace() usage. Cleaned this up - many thanks to everyone who reported this one.
|
|
||||||
- tibur found a bug in end tag parsing
|
|
||||||
|
|
||||||
|
|
||||||
2.6.2
|
|
||||||
- Switched over to VC 2010
|
|
||||||
- Fixed up all the build issues arising from that. (Lots of latent build problems.)
|
|
||||||
- Removed the old, now unmaintained and likely not working, build files.
|
|
||||||
- Fixed some static analysis issues reported by orbitcowboy from cppcheck.
|
|
||||||
- Bayard 95 sent in analysis from a different analyzer - fixes applied from that as well.
|
|
||||||
- Tim Kosse sent a patch fixing an infinite loop.
|
|
||||||
- Ma Anguo identified a doc issue.
|
|
||||||
- Eddie Cohen identified a missing qualifier resulting in a compilation error on some systems.
|
|
||||||
- Fixed a line ending bug. (What year is this? Can we all agree on a format for text files? Please? ...oh well.)
|
|
530
third_party/tinyxml/readme.txt
vendored
530
third_party/tinyxml/readme.txt
vendored
@ -1,530 +0,0 @@
|
|||||||
/** @mainpage
|
|
||||||
|
|
||||||
<h1> TinyXML </h1>
|
|
||||||
|
|
||||||
TinyXML is a simple, small, C++ XML parser that can be easily
|
|
||||||
integrated into other programs.
|
|
||||||
|
|
||||||
<h2> What it does. </h2>
|
|
||||||
|
|
||||||
In brief, TinyXML parses an XML document, and builds from that a
|
|
||||||
Document Object Model (DOM) that can be read, modified, and saved.
|
|
||||||
|
|
||||||
XML stands for "eXtensible Markup Language." It allows you to create
|
|
||||||
your own document markups. Where HTML does a very good job of marking
|
|
||||||
documents for browsers, XML allows you to define any kind of document
|
|
||||||
markup, for example a document that describes a "to do" list for an
|
|
||||||
organizer application. XML is a very structured and convenient format.
|
|
||||||
All those random file formats created to store application data can
|
|
||||||
all be replaced with XML. One parser for everything.
|
|
||||||
|
|
||||||
The best place for the complete, correct, and quite frankly hard to
|
|
||||||
read spec is at <a href="http://www.w3.org/TR/2004/REC-xml-20040204/">
|
|
||||||
http://www.w3.org/TR/2004/REC-xml-20040204/</a>. An intro to XML
|
|
||||||
(that I really like) can be found at
|
|
||||||
<a href="http://skew.org/xml/tutorial/">http://skew.org/xml/tutorial</a>.
|
|
||||||
|
|
||||||
There are different ways to access and interact with XML data.
|
|
||||||
TinyXML uses a Document Object Model (DOM), meaning the XML data is parsed
|
|
||||||
into a C++ objects that can be browsed and manipulated, and then
|
|
||||||
written to disk or another output stream. You can also construct an XML document
|
|
||||||
from scratch with C++ objects and write this to disk or another output
|
|
||||||
stream.
|
|
||||||
|
|
||||||
TinyXML is designed to be easy and fast to learn. It is two headers
|
|
||||||
and four cpp files. Simply add these to your project and off you go.
|
|
||||||
There is an example file - xmltest.cpp - to get you started.
|
|
||||||
|
|
||||||
TinyXML is released under the ZLib license,
|
|
||||||
so you can use it in open source or commercial code. The details
|
|
||||||
of the license are at the top of every source file.
|
|
||||||
|
|
||||||
TinyXML attempts to be a flexible parser, but with truly correct and
|
|
||||||
compliant XML output. TinyXML should compile on any reasonably C++
|
|
||||||
compliant system. It does not rely on exceptions or RTTI. It can be
|
|
||||||
compiled with or without STL support. TinyXML fully supports
|
|
||||||
the UTF-8 encoding, and the first 64k character entities.
|
|
||||||
|
|
||||||
|
|
||||||
<h2> What it doesn't do. </h2>
|
|
||||||
|
|
||||||
TinyXML doesn't parse or use DTDs (Document Type Definitions) or XSLs
|
|
||||||
(eXtensible Stylesheet Language.) There are other parsers out there
|
|
||||||
(check out www.sourceforge.org, search for XML) that are much more fully
|
|
||||||
featured. But they are also much bigger, take longer to set up in
|
|
||||||
your project, have a higher learning curve, and often have a more
|
|
||||||
restrictive license. If you are working with browsers or have more
|
|
||||||
complete XML needs, TinyXML is not the parser for you.
|
|
||||||
|
|
||||||
The following DTD syntax will not parse at this time in TinyXML:
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
<!DOCTYPE Archiv [
|
|
||||||
<!ELEMENT Comment (#PCDATA)>
|
|
||||||
]>
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
because TinyXML sees this as a !DOCTYPE node with an illegally
|
|
||||||
embedded !ELEMENT node. This may be addressed in the future.
|
|
||||||
|
|
||||||
<h2> Tutorials. </h2>
|
|
||||||
|
|
||||||
For the impatient, here is a tutorial to get you going. A great way to get started,
|
|
||||||
but it is worth your time to read this (very short) manual completely.
|
|
||||||
|
|
||||||
- @subpage tutorial0
|
|
||||||
|
|
||||||
<h2> Code Status. </h2>
|
|
||||||
|
|
||||||
TinyXML is mature, tested code. It is very stable. If you find
|
|
||||||
bugs, please file a bug report on the sourceforge web site
|
|
||||||
(www.sourceforge.net/projects/tinyxml). We'll get them straightened
|
|
||||||
out as soon as possible.
|
|
||||||
|
|
||||||
There are some areas of improvement; please check sourceforge if you are
|
|
||||||
interested in working on TinyXML.
|
|
||||||
|
|
||||||
<h2> Related Projects </h2>
|
|
||||||
|
|
||||||
TinyXML projects you may find useful! (Descriptions provided by the projects.)
|
|
||||||
|
|
||||||
<ul>
|
|
||||||
<li> <b>TinyXPath</b> (http://tinyxpath.sourceforge.net). TinyXPath is a small footprint
|
|
||||||
XPath syntax decoder, written in C++.</li>
|
|
||||||
<li> <b>TinyXML++</b> (http://code.google.com/p/ticpp/). TinyXML++ is a completely new
|
|
||||||
interface to TinyXML that uses MANY of the C++ strengths. Templates,
|
|
||||||
exceptions, and much better error handling.</li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<h2> Features </h2>
|
|
||||||
|
|
||||||
<h3> Using STL </h3>
|
|
||||||
|
|
||||||
TinyXML can be compiled to use or not use STL. When using STL, TinyXML
|
|
||||||
uses the std::string class, and fully supports std::istream, std::ostream,
|
|
||||||
operator<<, and operator>>. Many API methods have both 'const char*' and
|
|
||||||
'const std::string&' forms.
|
|
||||||
|
|
||||||
When STL support is compiled out, no STL files are included whatsoever. All
|
|
||||||
the string classes are implemented by TinyXML itself. API methods
|
|
||||||
all use the 'const char*' form for input.
|
|
||||||
|
|
||||||
Use the compile time #define:
|
|
||||||
|
|
||||||
TIXML_USE_STL
|
|
||||||
|
|
||||||
to compile one version or the other. This can be passed by the compiler,
|
|
||||||
or set as the first line of "tinyxml.h".
|
|
||||||
|
|
||||||
Note: If compiling the test code in Linux, setting the environment
|
|
||||||
variable TINYXML_USE_STL=YES/NO will control STL compilation. In the
|
|
||||||
Windows project file, STL and non STL targets are provided. In your project,
|
|
||||||
It's probably easiest to add the line "#define TIXML_USE_STL" as the first
|
|
||||||
line of tinyxml.h.
|
|
||||||
|
|
||||||
<h3> UTF-8 </h3>
|
|
||||||
|
|
||||||
TinyXML supports UTF-8 allowing to manipulate XML files in any language. TinyXML
|
|
||||||
also supports "legacy mode" - the encoding used before UTF-8 support and
|
|
||||||
probably best described as "extended ascii".
|
|
||||||
|
|
||||||
Normally, TinyXML will try to detect the correct encoding and use it. However,
|
|
||||||
by setting the value of TIXML_DEFAULT_ENCODING in the header file, TinyXML
|
|
||||||
can be forced to always use one encoding.
|
|
||||||
|
|
||||||
TinyXML will assume Legacy Mode until one of the following occurs:
|
|
||||||
<ol>
|
|
||||||
<li> If the non-standard but common "UTF-8 lead bytes" (0xef 0xbb 0xbf)
|
|
||||||
begin the file or data stream, TinyXML will read it as UTF-8. </li>
|
|
||||||
<li> If the declaration tag is read, and it has an encoding="UTF-8", then
|
|
||||||
TinyXML will read it as UTF-8. </li>
|
|
||||||
<li> If the declaration tag is read, and it has no encoding specified, then TinyXML will
|
|
||||||
read it as UTF-8. </li>
|
|
||||||
<li> If the declaration tag is read, and it has an encoding="something else", then TinyXML
|
|
||||||
will read it as Legacy Mode. In legacy mode, TinyXML will work as it did before. It's
|
|
||||||
not clear what that mode does exactly, but old content should keep working.</li>
|
|
||||||
<li> Until one of the above criteria is met, TinyXML runs in Legacy Mode.</li>
|
|
||||||
</ol>
|
|
||||||
|
|
||||||
What happens if the encoding is incorrectly set or detected? TinyXML will try
|
|
||||||
to read and pass through text seen as improperly encoded. You may get some strange results or
|
|
||||||
mangled characters. You may want to force TinyXML to the correct mode.
|
|
||||||
|
|
||||||
You may force TinyXML to Legacy Mode by using LoadFile( TIXML_ENCODING_LEGACY ) or
|
|
||||||
LoadFile( filename, TIXML_ENCODING_LEGACY ). You may force it to use legacy mode all
|
|
||||||
the time by setting TIXML_DEFAULT_ENCODING = TIXML_ENCODING_LEGACY. Likewise, you may
|
|
||||||
force it to TIXML_ENCODING_UTF8 with the same technique.
|
|
||||||
|
|
||||||
For English users, using English XML, UTF-8 is the same as low-ASCII. You
|
|
||||||
don't need to be aware of UTF-8 or change your code in any way. You can think
|
|
||||||
of UTF-8 as a "superset" of ASCII.
|
|
||||||
|
|
||||||
UTF-8 is not a double byte format - but it is a standard encoding of Unicode!
|
|
||||||
TinyXML does not use or directly support wchar, TCHAR, or Microsoft's _UNICODE at this time.
|
|
||||||
It is common to see the term "Unicode" improperly refer to UTF-16, a wide byte encoding
|
|
||||||
of unicode. This is a source of confusion.
|
|
||||||
|
|
||||||
For "high-ascii" languages - everything not English, pretty much - TinyXML can
|
|
||||||
handle all languages, at the same time, as long as the XML is encoded
|
|
||||||
in UTF-8. That can be a little tricky, older programs and operating systems
|
|
||||||
tend to use the "default" or "traditional" code page. Many apps (and almost all
|
|
||||||
modern ones) can output UTF-8, but older or stubborn (or just broken) ones
|
|
||||||
still output text in the default code page.
|
|
||||||
|
|
||||||
For example, Japanese systems traditionally use SHIFT-JIS encoding.
|
|
||||||
Text encoded as SHIFT-JIS can not be read by TinyXML.
|
|
||||||
A good text editor can import SHIFT-JIS and then save as UTF-8.
|
|
||||||
|
|
||||||
The <a href="http://skew.org/xml/tutorial/">Skew.org link</a> does a great
|
|
||||||
job covering the encoding issue.
|
|
||||||
|
|
||||||
The test file "utf8test.xml" is an XML containing English, Spanish, Russian,
|
|
||||||
and Simplified Chinese. (Hopefully they are translated correctly). The file
|
|
||||||
"utf8test.gif" is a screen capture of the XML file, rendered in IE. Note that
|
|
||||||
if you don't have the correct fonts (Simplified Chinese or Russian) on your
|
|
||||||
system, you won't see output that matches the GIF file even if you can parse
|
|
||||||
it correctly. Also note that (at least on my Windows machine) console output
|
|
||||||
is in a Western code page, so that Print() or printf() cannot correctly display
|
|
||||||
the file. This is not a bug in TinyXML - just an OS issue. No data is lost or
|
|
||||||
destroyed by TinyXML. The console just doesn't render UTF-8.
|
|
||||||
|
|
||||||
|
|
||||||
<h3> Entities </h3>
|
|
||||||
TinyXML recognizes the pre-defined "character entities", meaning special
|
|
||||||
characters. Namely:
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
& &
|
|
||||||
< <
|
|
||||||
> >
|
|
||||||
" "
|
|
||||||
' '
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
These are recognized when the XML document is read, and translated to there
|
|
||||||
UTF-8 equivalents. For instance, text with the XML of:
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
Far & Away
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
will have the Value() of "Far & Away" when queried from the TiXmlText object,
|
|
||||||
and will be written back to the XML stream/file as an ampersand. Older versions
|
|
||||||
of TinyXML "preserved" character entities, but the newer versions will translate
|
|
||||||
them into characters.
|
|
||||||
|
|
||||||
Additionally, any character can be specified by its Unicode code point:
|
|
||||||
The syntax " " or " " are both to the non-breaking space characher.
|
|
||||||
|
|
||||||
<h3> Printing </h3>
|
|
||||||
TinyXML can print output in several different ways that all have strengths and limitations.
|
|
||||||
|
|
||||||
- Print( FILE* ). Output to a std-C stream, which includes all C files as well as stdout.
|
|
||||||
- "Pretty prints", but you don't have control over printing options.
|
|
||||||
- The output is streamed directly to the FILE object, so there is no memory overhead
|
|
||||||
in the TinyXML code.
|
|
||||||
- used by Print() and SaveFile()
|
|
||||||
|
|
||||||
- operator<<. Output to a c++ stream.
|
|
||||||
- Integrates with standart C++ iostreams.
|
|
||||||
- Outputs in "network printing" mode without line breaks. Good for network transmission
|
|
||||||
and moving XML between C++ objects, but hard for a human to read.
|
|
||||||
|
|
||||||
- TiXmlPrinter. Output to a std::string or memory buffer.
|
|
||||||
- API is less concise
|
|
||||||
- Future printing options will be put here.
|
|
||||||
- Printing may change slightly in future versions as it is refined and expanded.
|
|
||||||
|
|
||||||
<h3> Streams </h3>
|
|
||||||
With TIXML_USE_STL on TinyXML supports C++ streams (operator <<,>>) streams as well
|
|
||||||
as C (FILE*) streams. There are some differences that you may need to be aware of.
|
|
||||||
|
|
||||||
C style output:
|
|
||||||
- based on FILE*
|
|
||||||
- the Print() and SaveFile() methods
|
|
||||||
|
|
||||||
Generates formatted output, with plenty of white space, intended to be as
|
|
||||||
human-readable as possible. They are very fast, and tolerant of ill formed
|
|
||||||
XML documents. For example, an XML document that contains 2 root elements
|
|
||||||
and 2 declarations, will still print.
|
|
||||||
|
|
||||||
C style input:
|
|
||||||
- based on FILE*
|
|
||||||
- the Parse() and LoadFile() methods
|
|
||||||
|
|
||||||
A fast, tolerant read. Use whenever you don't need the C++ streams.
|
|
||||||
|
|
||||||
C++ style output:
|
|
||||||
- based on std::ostream
|
|
||||||
- operator<<
|
|
||||||
|
|
||||||
Generates condensed output, intended for network transmission rather than
|
|
||||||
readability. Depending on your system's implementation of the ostream class,
|
|
||||||
these may be somewhat slower. (Or may not.) Not tolerant of ill formed XML:
|
|
||||||
a document should contain the correct one root element. Additional root level
|
|
||||||
elements will not be streamed out.
|
|
||||||
|
|
||||||
C++ style input:
|
|
||||||
- based on std::istream
|
|
||||||
- operator>>
|
|
||||||
|
|
||||||
Reads XML from a stream, making it useful for network transmission. The tricky
|
|
||||||
part is knowing when the XML document is complete, since there will almost
|
|
||||||
certainly be other data in the stream. TinyXML will assume the XML data is
|
|
||||||
complete after it reads the root element. Put another way, documents that
|
|
||||||
are ill-constructed with more than one root element will not read correctly.
|
|
||||||
Also note that operator>> is somewhat slower than Parse, due to both
|
|
||||||
implementation of the STL and limitations of TinyXML.
|
|
||||||
|
|
||||||
<h3> White space </h3>
|
|
||||||
The world simply does not agree on whether white space should be kept, or condensed.
|
|
||||||
For example, pretend the '_' is a space, and look at "Hello____world". HTML, and
|
|
||||||
at least some XML parsers, will interpret this as "Hello_world". They condense white
|
|
||||||
space. Some XML parsers do not, and will leave it as "Hello____world". (Remember
|
|
||||||
to keep pretending the _ is a space.) Others suggest that __Hello___world__ should become
|
|
||||||
Hello___world.
|
|
||||||
|
|
||||||
It's an issue that hasn't been resolved to my satisfaction. TinyXML supports the
|
|
||||||
first 2 approaches. Call TiXmlBase::SetCondenseWhiteSpace( bool ) to set the desired behavior.
|
|
||||||
The default is to condense white space.
|
|
||||||
|
|
||||||
If you change the default, you should call TiXmlBase::SetCondenseWhiteSpace( bool )
|
|
||||||
before making any calls to Parse XML data, and I don't recommend changing it after
|
|
||||||
it has been set.
|
|
||||||
|
|
||||||
|
|
||||||
<h3> Handles </h3>
|
|
||||||
|
|
||||||
Where browsing an XML document in a robust way, it is important to check
|
|
||||||
for null returns from method calls. An error safe implementation can
|
|
||||||
generate a lot of code like:
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
TiXmlElement* root = document.FirstChildElement( "Document" );
|
|
||||||
if ( root )
|
|
||||||
{
|
|
||||||
TiXmlElement* element = root->FirstChildElement( "Element" );
|
|
||||||
if ( element )
|
|
||||||
{
|
|
||||||
TiXmlElement* child = element->FirstChildElement( "Child" );
|
|
||||||
if ( child )
|
|
||||||
{
|
|
||||||
TiXmlElement* child2 = child->NextSiblingElement( "Child" );
|
|
||||||
if ( child2 )
|
|
||||||
{
|
|
||||||
// Finally do something useful.
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
Handles have been introduced to clean this up. Using the TiXmlHandle class,
|
|
||||||
the previous code reduces to:
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
TiXmlHandle docHandle( &document );
|
|
||||||
TiXmlElement* child2 = docHandle.FirstChild( "Document" ).FirstChild( "Element" ).Child( "Child", 1 ).ToElement();
|
|
||||||
if ( child2 )
|
|
||||||
{
|
|
||||||
// do something useful
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
Which is much easier to deal with. See TiXmlHandle for more information.
|
|
||||||
|
|
||||||
|
|
||||||
<h3> Row and Column tracking </h3>
|
|
||||||
Being able to track nodes and attributes back to their origin location
|
|
||||||
in source files can be very important for some applications. Additionally,
|
|
||||||
knowing where parsing errors occured in the original source can be very
|
|
||||||
time saving.
|
|
||||||
|
|
||||||
TinyXML can tracks the row and column origin of all nodes and attributes
|
|
||||||
in a text file. The TiXmlBase::Row() and TiXmlBase::Column() methods return
|
|
||||||
the origin of the node in the source text. The correct tabs can be
|
|
||||||
configured in TiXmlDocument::SetTabSize().
|
|
||||||
|
|
||||||
|
|
||||||
<h2> Using and Installing </h2>
|
|
||||||
|
|
||||||
To Compile and Run xmltest:
|
|
||||||
|
|
||||||
A Linux Makefile and a Windows Visual C++ .dsw file is provided.
|
|
||||||
Simply compile and run. It will write the file demotest.xml to your
|
|
||||||
disk and generate output on the screen. It also tests walking the
|
|
||||||
DOM by printing out the number of nodes found using different
|
|
||||||
techniques.
|
|
||||||
|
|
||||||
The Linux makefile is very generic and runs on many systems - it
|
|
||||||
is currently tested on mingw and
|
|
||||||
MacOSX. You do not need to run 'make depend'. The dependecies have been
|
|
||||||
hard coded.
|
|
||||||
|
|
||||||
<h3>Windows project file for VC6</h3>
|
|
||||||
<ul>
|
|
||||||
<li>tinyxml: tinyxml library, non-STL </li>
|
|
||||||
<li>tinyxmlSTL: tinyxml library, STL </li>
|
|
||||||
<li>tinyXmlTest: test app, non-STL </li>
|
|
||||||
<li>tinyXmlTestSTL: test app, STL </li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
<h3>Makefile</h3>
|
|
||||||
At the top of the makefile you can set:
|
|
||||||
|
|
||||||
PROFILE, DEBUG, and TINYXML_USE_STL. Details (such that they are) are in
|
|
||||||
the makefile.
|
|
||||||
|
|
||||||
In the tinyxml directory, type "make clean" then "make". The executable
|
|
||||||
file 'xmltest' will be created.
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<h3>To Use in an Application:</h3>
|
|
||||||
|
|
||||||
Add tinyxml.cpp, tinyxml.h, tinyxmlerror.cpp, tinyxmlparser.cpp, tinystr.cpp, and tinystr.h to your
|
|
||||||
project or make file. That's it! It should compile on any reasonably
|
|
||||||
compliant C++ system. You do not need to enable exceptions or
|
|
||||||
RTTI for TinyXML.
|
|
||||||
|
|
||||||
|
|
||||||
<h2> How TinyXML works. </h2>
|
|
||||||
|
|
||||||
An example is probably the best way to go. Take:
|
|
||||||
@verbatim
|
|
||||||
<?xml version="1.0" standalone=no>
|
|
||||||
<!-- Our to do list data -->
|
|
||||||
<ToDo>
|
|
||||||
<Item priority="1"> Go to the <bold>Toy store!</bold></Item>
|
|
||||||
<Item priority="2"> Do bills</Item>
|
|
||||||
</ToDo>
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
Its not much of a To Do list, but it will do. To read this file
|
|
||||||
(say "demo.xml") you would create a document, and parse it in:
|
|
||||||
@verbatim
|
|
||||||
TiXmlDocument doc( "demo.xml" );
|
|
||||||
doc.LoadFile();
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
And its ready to go. Now lets look at some lines and how they
|
|
||||||
relate to the DOM.
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
<?xml version="1.0" standalone=no>
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
The first line is a declaration, and gets turned into the
|
|
||||||
TiXmlDeclaration class. It will be the first child of the
|
|
||||||
document node.
|
|
||||||
|
|
||||||
This is the only directive/special tag parsed by TinyXML.
|
|
||||||
Generally directive tags are stored in TiXmlUnknown so the
|
|
||||||
commands wont be lost when it is saved back to disk.
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
<!-- Our to do list data -->
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
A comment. Will become a TiXmlComment object.
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
<ToDo>
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
The "ToDo" tag defines a TiXmlElement object. This one does not have
|
|
||||||
any attributes, but does contain 2 other elements.
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
<Item priority="1">
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
Creates another TiXmlElement which is a child of the "ToDo" element.
|
|
||||||
This element has 1 attribute, with the name "priority" and the value
|
|
||||||
"1".
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
Go to the
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
A TiXmlText. This is a leaf node and cannot contain other nodes.
|
|
||||||
It is a child of the "Item" TiXmlElement.
|
|
||||||
|
|
||||||
@verbatim
|
|
||||||
<bold>
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
|
|
||||||
Another TiXmlElement, this one a child of the "Item" element.
|
|
||||||
|
|
||||||
Etc.
|
|
||||||
|
|
||||||
Looking at the entire object tree, you end up with:
|
|
||||||
@verbatim
|
|
||||||
TiXmlDocument "demo.xml"
|
|
||||||
TiXmlDeclaration "version='1.0'" "standalone=no"
|
|
||||||
TiXmlComment " Our to do list data"
|
|
||||||
TiXmlElement "ToDo"
|
|
||||||
TiXmlElement "Item" Attribtutes: priority = 1
|
|
||||||
TiXmlText "Go to the "
|
|
||||||
TiXmlElement "bold"
|
|
||||||
TiXmlText "Toy store!"
|
|
||||||
TiXmlElement "Item" Attributes: priority=2
|
|
||||||
TiXmlText "Do bills"
|
|
||||||
@endverbatim
|
|
||||||
|
|
||||||
<h2> Documentation </h2>
|
|
||||||
|
|
||||||
The documentation is build with Doxygen, using the 'dox'
|
|
||||||
configuration file.
|
|
||||||
|
|
||||||
<h2> License </h2>
|
|
||||||
|
|
||||||
TinyXML is released under the zlib license:
|
|
||||||
|
|
||||||
This software is provided 'as-is', without any express or implied
|
|
||||||
warranty. In no event will the authors be held liable for any
|
|
||||||
damages arising from the use of this software.
|
|
||||||
|
|
||||||
Permission is granted to anyone to use this software for any
|
|
||||||
purpose, including commercial applications, and to alter it and
|
|
||||||
redistribute it freely, subject to the following restrictions:
|
|
||||||
|
|
||||||
1. The origin of this software must not be misrepresented; you must
|
|
||||||
not claim that you wrote the original software. If you use this
|
|
||||||
software in a product, an acknowledgment in the product documentation
|
|
||||||
would be appreciated but is not required.
|
|
||||||
|
|
||||||
2. Altered source versions must be plainly marked as such, and
|
|
||||||
must not be misrepresented as being the original software.
|
|
||||||
|
|
||||||
3. This notice may not be removed or altered from any source
|
|
||||||
distribution.
|
|
||||||
|
|
||||||
<h2> References </h2>
|
|
||||||
|
|
||||||
The World Wide Web Consortium is the definitive standard body for
|
|
||||||
XML, and their web pages contain huge amounts of information.
|
|
||||||
|
|
||||||
The definitive spec: <a href="http://www.w3.org/TR/2004/REC-xml-20040204/">
|
|
||||||
http://www.w3.org/TR/2004/REC-xml-20040204/</a>
|
|
||||||
|
|
||||||
I also recommend "XML Pocket Reference" by Robert Eckstein and published by
|
|
||||||
OReilly...the book that got the whole thing started.
|
|
||||||
|
|
||||||
<h2> Contributors, Contacts, and a Brief History </h2>
|
|
||||||
|
|
||||||
Thanks very much to everyone who sends suggestions, bugs, ideas, and
|
|
||||||
encouragement. It all helps, and makes this project fun. A special thanks
|
|
||||||
to the contributors on the web pages that keep it lively.
|
|
||||||
|
|
||||||
So many people have sent in bugs and ideas, that rather than list here
|
|
||||||
we try to give credit due in the "changes.txt" file.
|
|
||||||
|
|
||||||
TinyXML was originally written by Lee Thomason. (Often the "I" still
|
|
||||||
in the documentation.) Lee reviews changes and releases new versions,
|
|
||||||
with the help of Yves Berquin, Andrew Ellerton, and the tinyXml community.
|
|
||||||
|
|
||||||
We appreciate your suggestions, and would love to know if you
|
|
||||||
use TinyXML. Hopefully you will enjoy it and find it useful.
|
|
||||||
Please post questions, comments, file bugs, or contact us at:
|
|
||||||
|
|
||||||
www.sourceforge.net/projects/tinyxml
|
|
||||||
|
|
||||||
Lee Thomason, Yves Berquin, Andrew Ellerton
|
|
||||||
*/
|
|
111
third_party/tinyxml/tinystr.cpp
vendored
111
third_party/tinyxml/tinystr.cpp
vendored
@ -1,111 +0,0 @@
|
|||||||
/*
|
|
||||||
www.sourceforge.net/projects/tinyxml
|
|
||||||
|
|
||||||
This software is provided 'as-is', without any express or implied
|
|
||||||
warranty. In no event will the authors be held liable for any
|
|
||||||
damages arising from the use of this software.
|
|
||||||
|
|
||||||
Permission is granted to anyone to use this software for any
|
|
||||||
purpose, including commercial applications, and to alter it and
|
|
||||||
redistribute it freely, subject to the following restrictions:
|
|
||||||
|
|
||||||
1. The origin of this software must not be misrepresented; you must
|
|
||||||
not claim that you wrote the original software. If you use this
|
|
||||||
software in a product, an acknowledgment in the product documentation
|
|
||||||
would be appreciated but is not required.
|
|
||||||
|
|
||||||
2. Altered source versions must be plainly marked as such, and
|
|
||||||
must not be misrepresented as being the original software.
|
|
||||||
|
|
||||||
3. This notice may not be removed or altered from any source
|
|
||||||
distribution.
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef TIXML_USE_STL
|
|
||||||
|
|
||||||
#include "tinystr.h"
|
|
||||||
|
|
||||||
// Error value for find primitive
|
|
||||||
const TiXmlString::size_type TiXmlString::npos = static_cast< TiXmlString::size_type >(-1);
|
|
||||||
|
|
||||||
|
|
||||||
// Null rep.
|
|
||||||
TiXmlString::Rep TiXmlString::nullrep_ = { 0, 0, { '\0' } };
|
|
||||||
|
|
||||||
|
|
||||||
void TiXmlString::reserve (size_type cap)
|
|
||||||
{
|
|
||||||
if (cap > capacity())
|
|
||||||
{
|
|
||||||
TiXmlString tmp;
|
|
||||||
tmp.init(length(), cap);
|
|
||||||
memcpy(tmp.start(), data(), length());
|
|
||||||
swap(tmp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
TiXmlString& TiXmlString::assign(const char* str, size_type len)
|
|
||||||
{
|
|
||||||
size_type cap = capacity();
|
|
||||||
if (len > cap || cap > 3*(len + 8))
|
|
||||||
{
|
|
||||||
TiXmlString tmp;
|
|
||||||
tmp.init(len);
|
|
||||||
memcpy(tmp.start(), str, len);
|
|
||||||
swap(tmp);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
memmove(start(), str, len);
|
|
||||||
set_size(len);
|
|
||||||
}
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
TiXmlString& TiXmlString::append(const char* str, size_type len)
|
|
||||||
{
|
|
||||||
size_type newsize = length() + len;
|
|
||||||
if (newsize > capacity())
|
|
||||||
{
|
|
||||||
reserve (newsize + capacity());
|
|
||||||
}
|
|
||||||
memmove(finish(), str, len);
|
|
||||||
set_size(newsize);
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
TiXmlString operator + (const TiXmlString & a, const TiXmlString & b)
|
|
||||||
{
|
|
||||||
TiXmlString tmp;
|
|
||||||
tmp.reserve(a.length() + b.length());
|
|
||||||
tmp += a;
|
|
||||||
tmp += b;
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
TiXmlString operator + (const TiXmlString & a, const char* b)
|
|
||||||
{
|
|
||||||
TiXmlString tmp;
|
|
||||||
TiXmlString::size_type b_len = static_cast<TiXmlString::size_type>( strlen(b) );
|
|
||||||
tmp.reserve(a.length() + b_len);
|
|
||||||
tmp += a;
|
|
||||||
tmp.append(b, b_len);
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
TiXmlString operator + (const char* a, const TiXmlString & b)
|
|
||||||
{
|
|
||||||
TiXmlString tmp;
|
|
||||||
TiXmlString::size_type a_len = static_cast<TiXmlString::size_type>( strlen(a) );
|
|
||||||
tmp.reserve(a_len + b.length());
|
|
||||||
tmp.append(a, a_len);
|
|
||||||
tmp += b;
|
|
||||||
return tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#endif // TIXML_USE_STL
|
|
305
third_party/tinyxml/tinystr.h
vendored
305
third_party/tinyxml/tinystr.h
vendored
@ -1,305 +0,0 @@
|
|||||||
/*
|
|
||||||
www.sourceforge.net/projects/tinyxml
|
|
||||||
|
|
||||||
This software is provided 'as-is', without any express or implied
|
|
||||||
warranty. In no event will the authors be held liable for any
|
|
||||||
damages arising from the use of this software.
|
|
||||||
|
|
||||||
Permission is granted to anyone to use this software for any
|
|
||||||
purpose, including commercial applications, and to alter it and
|
|
||||||
redistribute it freely, subject to the following restrictions:
|
|
||||||
|
|
||||||
1. The origin of this software must not be misrepresented; you must
|
|
||||||
not claim that you wrote the original software. If you use this
|
|
||||||
software in a product, an acknowledgment in the product documentation
|
|
||||||
would be appreciated but is not required.
|
|
||||||
|
|
||||||
2. Altered source versions must be plainly marked as such, and
|
|
||||||
must not be misrepresented as being the original software.
|
|
||||||
|
|
||||||
3. This notice may not be removed or altered from any source
|
|
||||||
distribution.
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef TIXML_USE_STL
|
|
||||||
|
|
||||||
#ifndef TIXML_STRING_INCLUDED
|
|
||||||
#define TIXML_STRING_INCLUDED
|
|
||||||
|
|
||||||
#include <assert.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
/* The support for explicit isn't that universal, and it isn't really
|
|
||||||
required - it is used to check that the TiXmlString class isn't incorrectly
|
|
||||||
used. Be nice to old compilers and macro it here:
|
|
||||||
*/
|
|
||||||
#if defined(_MSC_VER) && (_MSC_VER >= 1200 )
|
|
||||||
// Microsoft visual studio, version 6 and higher.
|
|
||||||
#define TIXML_EXPLICIT explicit
|
|
||||||
#elif defined(__GNUC__) && (__GNUC__ >= 3 )
|
|
||||||
// GCC version 3 and higher.s
|
|
||||||
#define TIXML_EXPLICIT explicit
|
|
||||||
#else
|
|
||||||
#define TIXML_EXPLICIT
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
TiXmlString is an emulation of a subset of the std::string template.
|
|
||||||
Its purpose is to allow compiling TinyXML on compilers with no or poor STL support.
|
|
||||||
Only the member functions relevant to the TinyXML project have been implemented.
|
|
||||||
The buffer allocation is made by a simplistic power of 2 like mechanism : if we increase
|
|
||||||
a string and there's no more room, we allocate a buffer twice as big as we need.
|
|
||||||
*/
|
|
||||||
class TiXmlString
|
|
||||||
{
|
|
||||||
public :
|
|
||||||
// The size type used
|
|
||||||
typedef size_t size_type;
|
|
||||||
|
|
||||||
// Error value for find primitive
|
|
||||||
static const size_type npos; // = -1;
|
|
||||||
|
|
||||||
|
|
||||||
// TiXmlString empty constructor
|
|
||||||
TiXmlString () : rep_(&nullrep_)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
// TiXmlString copy constructor
|
|
||||||
TiXmlString ( const TiXmlString & copy) : rep_(0)
|
|
||||||
{
|
|
||||||
init(copy.length());
|
|
||||||
memcpy(start(), copy.data(), length());
|
|
||||||
}
|
|
||||||
|
|
||||||
// TiXmlString constructor, based on a string
|
|
||||||
TIXML_EXPLICIT TiXmlString ( const char * copy) : rep_(0)
|
|
||||||
{
|
|
||||||
init( static_cast<size_type>( strlen(copy) ));
|
|
||||||
memcpy(start(), copy, length());
|
|
||||||
}
|
|
||||||
|
|
||||||
// TiXmlString constructor, based on a string
|
|
||||||
TIXML_EXPLICIT TiXmlString ( const char * str, size_type len) : rep_(0)
|
|
||||||
{
|
|
||||||
init(len);
|
|
||||||
memcpy(start(), str, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
// TiXmlString destructor
|
|
||||||
~TiXmlString ()
|
|
||||||
{
|
|
||||||
quit();
|
|
||||||
}
|
|
||||||
|
|
||||||
TiXmlString& operator = (const char * copy)
|
|
||||||
{
|
|
||||||
return assign( copy, (size_type)strlen(copy));
|
|
||||||
}
|
|
||||||
|
|
||||||
TiXmlString& operator = (const TiXmlString & copy)
|
|
||||||
{
|
|
||||||
return assign(copy.start(), copy.length());
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// += operator. Maps to append
|
|
||||||
TiXmlString& operator += (const char * suffix)
|
|
||||||
{
|
|
||||||
return append(suffix, static_cast<size_type>( strlen(suffix) ));
|
|
||||||
}
|
|
||||||
|
|
||||||
// += operator. Maps to append
|
|
||||||
TiXmlString& operator += (char single)
|
|
||||||
{
|
|
||||||
return append(&single, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
// += operator. Maps to append
|
|
||||||
TiXmlString& operator += (const TiXmlString & suffix)
|
|
||||||
{
|
|
||||||
return append(suffix.data(), suffix.length());
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Convert a TiXmlString into a null-terminated char *
|
|
||||||
const char * c_str () const { return rep_->str; }
|
|
||||||
|
|
||||||
// Convert a TiXmlString into a char * (need not be null terminated).
|
|
||||||
const char * data () const { return rep_->str; }
|
|
||||||
|
|
||||||
// Return the length of a TiXmlString
|
|
||||||
size_type length () const { return rep_->size; }
|
|
||||||
|
|
||||||
// Alias for length()
|
|
||||||
size_type size () const { return rep_->size; }
|
|
||||||
|
|
||||||
// Checks if a TiXmlString is empty
|
|
||||||
bool empty () const { return rep_->size == 0; }
|
|
||||||
|
|
||||||
// Return capacity of string
|
|
||||||
size_type capacity () const { return rep_->capacity; }
|
|
||||||
|
|
||||||
|
|
||||||
// single char extraction
|
|
||||||
const char& at (size_type index) const
|
|
||||||
{
|
|
||||||
assert( index < length() );
|
|
||||||
return rep_->str[ index ];
|
|
||||||
}
|
|
||||||
|
|
||||||
// [] operator
|
|
||||||
char& operator [] (size_type index) const
|
|
||||||
{
|
|
||||||
assert( index < length() );
|
|
||||||
return rep_->str[ index ];
|
|
||||||
}
|
|
||||||
|
|
||||||
// find a char in a string. Return TiXmlString::npos if not found
|
|
||||||
size_type find (char lookup) const
|
|
||||||
{
|
|
||||||
return find(lookup, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// find a char in a string from an offset. Return TiXmlString::npos if not found
|
|
||||||
size_type find (char tofind, size_type offset) const
|
|
||||||
{
|
|
||||||
if (offset >= length()) return npos;
|
|
||||||
|
|
||||||
for (const char* p = c_str() + offset; *p != '\0'; ++p)
|
|
||||||
{
|
|
||||||
if (*p == tofind) return static_cast< size_type >( p - c_str() );
|
|
||||||
}
|
|
||||||
return npos;
|
|
||||||
}
|
|
||||||
|
|
||||||
void clear ()
|
|
||||||
{
|
|
||||||
//Lee:
|
|
||||||
//The original was just too strange, though correct:
|
|
||||||
// TiXmlString().swap(*this);
|
|
||||||
//Instead use the quit & re-init:
|
|
||||||
quit();
|
|
||||||
init(0,0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Function to reserve a big amount of data when we know we'll need it. Be aware that this
|
|
||||||
function DOES NOT clear the content of the TiXmlString if any exists.
|
|
||||||
*/
|
|
||||||
void reserve (size_type cap);
|
|
||||||
|
|
||||||
TiXmlString& assign (const char* str, size_type len);
|
|
||||||
|
|
||||||
TiXmlString& append (const char* str, size_type len);
|
|
||||||
|
|
||||||
void swap (TiXmlString& other)
|
|
||||||
{
|
|
||||||
Rep* r = rep_;
|
|
||||||
rep_ = other.rep_;
|
|
||||||
other.rep_ = r;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
|
|
||||||
void init(size_type sz) { init(sz, sz); }
|
|
||||||
void set_size(size_type sz) { rep_->str[ rep_->size = sz ] = '\0'; }
|
|
||||||
char* start() const { return rep_->str; }
|
|
||||||
char* finish() const { return rep_->str + rep_->size; }
|
|
||||||
|
|
||||||
struct Rep
|
|
||||||
{
|
|
||||||
size_type size, capacity;
|
|
||||||
char str[1];
|
|
||||||
};
|
|
||||||
|
|
||||||
void init(size_type sz, size_type cap)
|
|
||||||
{
|
|
||||||
if (cap)
|
|
||||||
{
|
|
||||||
// Lee: the original form:
|
|
||||||
// rep_ = static_cast<Rep*>(operator new(sizeof(Rep) + cap));
|
|
||||||
// doesn't work in some cases of new being overloaded. Switching
|
|
||||||
// to the normal allocation, although use an 'int' for systems
|
|
||||||
// that are overly picky about structure alignment.
|
|
||||||
const size_type bytesNeeded = sizeof(Rep) + cap;
|
|
||||||
const size_type intsNeeded = ( bytesNeeded + sizeof(int) - 1 ) / sizeof( int );
|
|
||||||
rep_ = reinterpret_cast<Rep*>( new int[ intsNeeded ] );
|
|
||||||
|
|
||||||
rep_->str[ rep_->size = sz ] = '\0';
|
|
||||||
rep_->capacity = cap;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
rep_ = &nullrep_;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void quit()
|
|
||||||
{
|
|
||||||
if (rep_ != &nullrep_)
|
|
||||||
{
|
|
||||||
// The rep_ is really an array of ints. (see the allocator, above).
|
|
||||||
// Cast it back before delete, so the compiler won't incorrectly call destructors.
|
|
||||||
delete [] ( reinterpret_cast<int*>( rep_ ) );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Rep * rep_;
|
|
||||||
static Rep nullrep_;
|
|
||||||
|
|
||||||
} ;
|
|
||||||
|
|
||||||
|
|
||||||
inline bool operator == (const TiXmlString & a, const TiXmlString & b)
|
|
||||||
{
|
|
||||||
return ( a.length() == b.length() ) // optimization on some platforms
|
|
||||||
&& ( strcmp(a.c_str(), b.c_str()) == 0 ); // actual compare
|
|
||||||
}
|
|
||||||
inline bool operator < (const TiXmlString & a, const TiXmlString & b)
|
|
||||||
{
|
|
||||||
return strcmp(a.c_str(), b.c_str()) < 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
inline bool operator != (const TiXmlString & a, const TiXmlString & b) { return !(a == b); }
|
|
||||||
inline bool operator > (const TiXmlString & a, const TiXmlString & b) { return b < a; }
|
|
||||||
inline bool operator <= (const TiXmlString & a, const TiXmlString & b) { return !(b < a); }
|
|
||||||
inline bool operator >= (const TiXmlString & a, const TiXmlString & b) { return !(a < b); }
|
|
||||||
|
|
||||||
inline bool operator == (const TiXmlString & a, const char* b) { return strcmp(a.c_str(), b) == 0; }
|
|
||||||
inline bool operator == (const char* a, const TiXmlString & b) { return b == a; }
|
|
||||||
inline bool operator != (const TiXmlString & a, const char* b) { return !(a == b); }
|
|
||||||
inline bool operator != (const char* a, const TiXmlString & b) { return !(b == a); }
|
|
||||||
|
|
||||||
TiXmlString operator + (const TiXmlString & a, const TiXmlString & b);
|
|
||||||
TiXmlString operator + (const TiXmlString & a, const char* b);
|
|
||||||
TiXmlString operator + (const char* a, const TiXmlString & b);
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
TiXmlOutStream is an emulation of std::ostream. It is based on TiXmlString.
|
|
||||||
Only the operators that we need for TinyXML have been developped.
|
|
||||||
*/
|
|
||||||
class TiXmlOutStream : public TiXmlString
|
|
||||||
{
|
|
||||||
public :
|
|
||||||
|
|
||||||
// TiXmlOutStream << operator.
|
|
||||||
TiXmlOutStream & operator << (const TiXmlString & in)
|
|
||||||
{
|
|
||||||
*this += in;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TiXmlOutStream << operator.
|
|
||||||
TiXmlOutStream & operator << (const char * in)
|
|
||||||
{
|
|
||||||
*this += in;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
} ;
|
|
||||||
|
|
||||||
#endif // TIXML_STRING_INCLUDED
|
|
||||||
#endif // TIXML_USE_STL
|
|
1885
third_party/tinyxml/tinyxml.cpp
vendored
1885
third_party/tinyxml/tinyxml.cpp
vendored
File diff suppressed because it is too large
Load Diff
1805
third_party/tinyxml/tinyxml.h
vendored
1805
third_party/tinyxml/tinyxml.h
vendored
File diff suppressed because it is too large
Load Diff
52
third_party/tinyxml/tinyxmlerror.cpp
vendored
52
third_party/tinyxml/tinyxmlerror.cpp
vendored
@ -1,52 +0,0 @@
|
|||||||
/*
|
|
||||||
www.sourceforge.net/projects/tinyxml
|
|
||||||
Original code (2.0 and earlier )copyright (c) 2000-2006 Lee Thomason (www.grinninglizard.com)
|
|
||||||
|
|
||||||
This software is provided 'as-is', without any express or implied
|
|
||||||
warranty. In no event will the authors be held liable for any
|
|
||||||
damages arising from the use of this software.
|
|
||||||
|
|
||||||
Permission is granted to anyone to use this software for any
|
|
||||||
purpose, including commercial applications, and to alter it and
|
|
||||||
redistribute it freely, subject to the following restrictions:
|
|
||||||
|
|
||||||
1. The origin of this software must not be misrepresented; you must
|
|
||||||
not claim that you wrote the original software. If you use this
|
|
||||||
software in a product, an acknowledgment in the product documentation
|
|
||||||
would be appreciated but is not required.
|
|
||||||
|
|
||||||
2. Altered source versions must be plainly marked as such, and
|
|
||||||
must not be misrepresented as being the original software.
|
|
||||||
|
|
||||||
3. This notice may not be removed or altered from any source
|
|
||||||
distribution.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "tinyxml.h"
|
|
||||||
|
|
||||||
// The goal of the seperate error file is to make the first
|
|
||||||
// step towards localization. tinyxml (currently) only supports
|
|
||||||
// english error messages, but the could now be translated.
|
|
||||||
//
|
|
||||||
// It also cleans up the code a bit.
|
|
||||||
//
|
|
||||||
|
|
||||||
const char* TiXmlBase::errorString[ TiXmlBase::TIXML_ERROR_STRING_COUNT ] =
|
|
||||||
{
|
|
||||||
"No error",
|
|
||||||
"Error",
|
|
||||||
"Failed to open file",
|
|
||||||
"Error parsing Element.",
|
|
||||||
"Failed to read Element name",
|
|
||||||
"Error reading Element value.",
|
|
||||||
"Error reading Attributes.",
|
|
||||||
"Error: empty tag.",
|
|
||||||
"Error reading end tag.",
|
|
||||||
"Error parsing Unknown.",
|
|
||||||
"Error parsing Comment.",
|
|
||||||
"Error parsing Declaration.",
|
|
||||||
"Error document empty.",
|
|
||||||
"Error null (0) or unexpected EOF found in input stream.",
|
|
||||||
"Error parsing CDATA.",
|
|
||||||
"Error when TiXmlDocument added to document, because TiXmlDocument can only be at the root.",
|
|
||||||
};
|
|
1637
third_party/tinyxml/tinyxmlparser.cpp
vendored
1637
third_party/tinyxml/tinyxmlparser.cpp
vendored
File diff suppressed because it is too large
Load Diff
1
third_party/tinyxml2
vendored
Submodule
1
third_party/tinyxml2
vendored
Submodule
@ -0,0 +1 @@
|
|||||||
|
Subproject commit 312a8092245df393db14a0b2427457ed2ba75e1b
|
Loading…
x
Reference in New Issue
Block a user