diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/ui/CMakeLists.txt | 1 | ||||
-rw-r--r-- | src/ui/ThemeManager.cpp | 62 | ||||
-rw-r--r-- | src/ui/ThemeResourceDictionary.cpp | 57 |
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 |