aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/ui/CMakeLists.txt1
-rw-r--r--src/ui/ThemeManager.cpp62
-rw-r--r--src/ui/ThemeResourceDictionary.cpp57
3 files changed, 79 insertions, 41 deletions
diff --git a/src/ui/CMakeLists.txt b/src/ui/CMakeLists.txt
index 56205db0..91128862 100644
--- a/src/ui/CMakeLists.txt
+++ b/src/ui/CMakeLists.txt
@@ -1,6 +1,7 @@
add_library(cru_ui SHARED
Helper.cpp
ThemeManager.cpp
+ ThemeResourceDictionary.cpp
components/Component.cpp
components/Menu.cpp
components/PopupButton.cpp
diff --git a/src/ui/ThemeManager.cpp b/src/ui/ThemeManager.cpp
index 4649f70f..1adbddaf 100644
--- a/src/ui/ThemeManager.cpp
+++ b/src/ui/ThemeManager.cpp
@@ -7,6 +7,7 @@
#include "cru/platform/graphics/Brush.h"
#include "cru/platform/graphics/Factory.h"
#include "cru/platform/gui/UiApplication.h"
+#include "cru/ui/ThemeResourceDictionary.h"
#include "cru/ui/style/StyleRuleSet.h"
#include "cru/xml/XmlParser.h"
@@ -24,11 +25,30 @@ ThemeManager::ThemeManager() {
throw Exception(u"Default resources file not found.");
}
- ReadResourcesFile(String::FromStdPath(resourses_file));
+ PrependThemeResourceDictionary(
+ ThemeResourceDictionary::FromFile(String::FromStdPath(resourses_file)));
}
ThemeManager::~ThemeManager() {}
+std::vector<ThemeResourceDictionary*>
+ThemeManager::GetThemeResourceDictionaryList() const {
+ std::vector<ThemeResourceDictionary*> result;
+ for (const auto& theme_resource_dictionary :
+ theme_resource_dictionary_list_) {
+ result.push_back(theme_resource_dictionary.get());
+ }
+ return result;
+}
+
+void ThemeManager::PrependThemeResourceDictionary(
+ std::unique_ptr<ThemeResourceDictionary> theme_resource_dictionary) {
+ theme_resource_dictionary_list_.insert(
+ theme_resource_dictionary_list_.begin(),
+ std::move(theme_resource_dictionary));
+ theme_resource_change_event_.Raise(nullptr);
+}
+
std::shared_ptr<platform::graphics::IBrush> ThemeManager::GetResourceBrush(
const String& key) {
return GetResource<std::shared_ptr<platform::graphics::IBrush>>(key);
@@ -43,44 +63,4 @@ std::shared_ptr<style::StyleRuleSet> ThemeManager::GetResourceStyleRuleSet(
const String& key) {
return GetResource<std::shared_ptr<style::StyleRuleSet>>(key);
}
-
-void ThemeManager::ReadResourcesFile(const String& file_path) {
- io::FileStream stream(file_path, io::OpenFileFlags::Read);
- auto xml_string = stream.ReadAllAsString();
- auto parser = xml::XmlParser(xml_string);
- SetThemeXml(parser.Parse());
-}
-
-void ThemeManager::SetThemeXml(xml::XmlElementNode* root) {
- theme_resource_xml_root_.reset(root);
- theme_resource_map_.clear();
-
- if (!theme_resource_xml_root_->GetTag().CaseInsensitiveEqual(u"Theme")) {
- throw Exception(u"Root tag of theme must be \"Theme\".");
- }
-
- for (auto child : theme_resource_xml_root_->GetChildren()) {
- if (child->IsElementNode()) {
- auto c = child->AsElement();
- if (c->GetTag().CaseInsensitiveEqual(u"Resource")) {
- auto key_attr = c->GetOptionalAttributeCaseInsensitive(u"key");
- if (!key_attr) {
- throw Exception(u"\"key\" attribute is required for resource.");
- }
- if (!c->GetChildElementCount()) {
- throw Exception(u"Resource must have only one child element.");
- }
-
- ResourceEntry entry;
-
- entry.name = *key_attr;
- entry.xml_node = c->GetFirstChildElement();
-
- theme_resource_map_[entry.name] = std::move(entry);
- }
- }
- }
-
- theme_resource_change_event_.Raise(nullptr);
-}
} // namespace cru::ui
diff --git a/src/ui/ThemeResourceDictionary.cpp b/src/ui/ThemeResourceDictionary.cpp
new file mode 100644
index 00000000..d9483f6c
--- /dev/null
+++ b/src/ui/ThemeResourceDictionary.cpp
@@ -0,0 +1,57 @@
+#include "cru/ui/ThemeResourceDictionary.h"
+#include "cru/common/io/FileStream.h"
+#include "cru/common/log/Logger.h"
+#include "cru/xml/XmlNode.h"
+#include "cru/xml/XmlParser.h"
+
+namespace cru::ui {
+
+std::unique_ptr<ThemeResourceDictionary> ThemeResourceDictionary::FromFile(
+ const String& file_path) {
+ io::FileStream stream(file_path, io::OpenFileFlags::Read);
+ auto xml_string = stream.ReadAllAsString();
+ auto parser = xml::XmlParser(xml_string);
+ return std::make_unique<ThemeResourceDictionary>(parser.Parse(), false);
+}
+
+ThemeResourceDictionary::ThemeResourceDictionary(xml::XmlElementNode* xml_root,
+ bool clone) {
+ Expects(xml_root);
+ xml_root_.reset(clone ? xml_root->Clone()->AsElement() : xml_root);
+ UpdateResourceMap(xml_root_.get());
+}
+
+ThemeResourceDictionary::~ThemeResourceDictionary() = default;
+
+void ThemeResourceDictionary::UpdateResourceMap(xml::XmlElementNode* xml_root) {
+ if (!xml_root->GetTag().CaseInsensitiveEqual(u"Theme")) {
+ throw Exception(u"Root tag of theme must be 'Theme'.");
+ }
+
+ for (auto child : xml_root->GetChildren()) {
+ if (child->IsElementNode()) {
+ auto c = child->AsElement();
+ if (c->GetTag().CaseInsensitiveEqual(u"Resource")) {
+ auto key_attr = c->GetOptionalAttributeCaseInsensitive(u"key");
+ if (!key_attr) {
+ throw Exception(u"'key' attribute is required for resource.");
+ }
+ if (c->GetChildElementCount() != 1) {
+ throw Exception(u"Resource must have only one child element.");
+ }
+
+ ResourceEntry entry;
+
+ entry.name = *key_attr;
+ entry.xml_node = c->GetFirstChildElement();
+
+ resource_map_[entry.name] = std::move(entry);
+ } else {
+ CRU_LOG_DEBUG(u"Ignore unknown element {} of theme.", c->GetTag());
+ }
+ } else {
+ CRU_LOG_WARN(u"Ignore text node of theme.");
+ }
+ }
+}
+} // namespace cru::ui