| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491 |
- /*
- * This source file is part of RmlUi, the HTML/CSS Interface Middleware
- *
- * For the latest information, see http://github.com/mikke89/RmlUi
- *
- * Copyright (c) 2018 Michael R. P. Ragazzon
- * Copyright (c) 2019-2023 The RmlUi Team, and contributors
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- * in the Software without restriction, including without limitation the rights
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the Software is
- * furnished to do so, subject to the following conditions:
- *
- * The above copyright notice and this permission notice shall be included in
- * all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- * THE SOFTWARE.
- *
- */
- #include <RmlUi/Core.h>
- #include <RmlUi/Core/StreamMemory.h>
- #include <RmlUi/Core/TransformPrimitive.h>
- #include <RmlUi/Debugger.h>
- #include <RmlUi_Backend.h>
- #include <Shell.h>
- static const Rml::String sandbox_default_rcss = R"(
- body { top: 0; left: 0; right: 0; bottom: 0; overflow: hidden auto; }
- scrollbarvertical { width: 15px; }
- scrollbarvertical slidertrack { background: #eee; }
- scrollbarvertical slidertrack:active { background: #ddd; }
- scrollbarvertical sliderbar { width: 15px; min-height: 30px; background: #aaa; }
- scrollbarvertical sliderbar:hover { background: #888; }
- scrollbarvertical sliderbar:active { background: #666; }
- scrollbarhorizontal { height: 15px; }
- scrollbarhorizontal slidertrack { background: #eee; }
- scrollbarhorizontal slidertrack:active { background: #ddd; }
- scrollbarhorizontal sliderbar { height: 15px; min-width: 30px; background: #aaa; }
- scrollbarhorizontal sliderbar:hover { background: #888; }
- scrollbarhorizontal sliderbar:active { background: #666; }
- )";
- class DemoWindow : public Rml::EventListener {
- public:
- DemoWindow(const Rml::String& title, Rml::Context* context)
- {
- using namespace Rml;
- document = context->LoadDocument("basic/demo/data/demo.rml");
- if (document)
- {
- document->GetElementById("title")->SetInnerRML(title);
- // Add sandbox default text.
- if (auto source = rmlui_dynamic_cast<Rml::ElementFormControl*>(document->GetElementById("sandbox_rml_source")))
- {
- auto value = source->GetValue();
- value += "<p>Write your RML here</p>\n\n<!-- <img src=\"assets/high_scores_alien_1.tga\"/> -->";
- source->SetValue(value);
- }
- // Prepare sandbox document.
- if (auto target = document->GetElementById("sandbox_target"))
- {
- iframe = context->CreateDocument();
- auto iframe_ptr = iframe->GetParentNode()->RemoveChild(iframe);
- target->AppendChild(std::move(iframe_ptr));
- iframe->SetProperty(PropertyId::Position, Property(Style::Position::Absolute));
- iframe->SetProperty(PropertyId::Display, Property(Style::Display::Block));
- iframe->SetInnerRML("<p>Rendered output goes here.</p>");
- // Load basic RML style sheet
- Rml::String style_sheet_content;
- {
- // Load file into string
- auto file_interface = Rml::GetFileInterface();
- Rml::FileHandle handle = file_interface->Open("assets/rml.rcss");
- size_t length = file_interface->Length(handle);
- style_sheet_content.resize(length);
- file_interface->Read((void*)style_sheet_content.data(), length, handle);
- file_interface->Close(handle);
- style_sheet_content += sandbox_default_rcss;
- }
- Rml::StreamMemory stream((Rml::byte*)style_sheet_content.data(), style_sheet_content.size());
- stream.SetSourceURL("sandbox://default_rcss");
- rml_basic_style_sheet = MakeShared<Rml::StyleSheetContainer>();
- rml_basic_style_sheet->LoadStyleSheetContainer(&stream);
- }
- // Add sandbox style sheet text.
- if (auto source = rmlui_dynamic_cast<Rml::ElementFormControl*>(document->GetElementById("sandbox_rcss_source")))
- {
- Rml::String value = "/* Write your RCSS here */\n\n/* body { color: #fea; background: #224; }\nimg { image-color: red; } */";
- source->SetValue(value);
- SetSandboxStylesheet(value);
- }
- gauge = document->GetElementById("gauge");
- progress_horizontal = document->GetElementById("progress_horizontal");
- document->Show();
- }
- }
- void Update()
- {
- if (iframe)
- {
- iframe->UpdateDocument();
- }
- if (submitting && gauge && progress_horizontal)
- {
- using namespace Rml;
- constexpr float progressbars_time = 2.f;
- const float progress = Math::Min(float(GetSystemInterface()->GetElapsedTime() - submitting_start_time) / progressbars_time, 2.f);
- float value_gauge = 1.0f;
- float value_horizontal = 0.0f;
- if (progress < 1.0f)
- value_gauge = 0.5f - 0.5f * Math::Cos(Math::RMLUI_PI * progress);
- else
- value_horizontal = 0.5f - 0.5f * Math::Cos(Math::RMLUI_PI * (progress - 1.0f));
- progress_horizontal->SetAttribute("value", value_horizontal);
- const float value_begin = 0.09f;
- const float value_end = 1.f - value_begin;
- float value_mapped = value_begin + value_gauge * (value_end - value_begin);
- gauge->SetAttribute("value", value_mapped);
- auto value_gauge_str = CreateString(10, "%d %%", Math::RoundToInteger(value_gauge * 100.f));
- auto value_horizontal_str = CreateString(10, "%d %%", Math::RoundToInteger(value_horizontal * 100.f));
- if (auto el_value = document->GetElementById("gauge_value"))
- el_value->SetInnerRML(value_gauge_str);
- if (auto el_value = document->GetElementById("progress_value"))
- el_value->SetInnerRML(value_horizontal_str);
- String label = "Placing tubes";
- size_t num_dots = (size_t(progress * 10.f) % 4);
- if (progress > 1.0f)
- label += "... Placed! Assembling message";
- if (progress < 2.0f)
- label += String(num_dots, '.');
- else
- label += "... Done!";
- if (auto el_label = document->GetElementById("progress_label"))
- el_label->SetInnerRML(label);
- if (progress >= 2.0f)
- {
- submitting = false;
- if (auto el_output = document->GetElementById("form_output"))
- el_output->SetInnerRML(submit_message);
- }
- document->GetContext()->RequestNextUpdate(.0);
- }
- }
- void Shutdown()
- {
- if (document)
- {
- document->Close();
- document = nullptr;
- }
- }
- void ProcessEvent(Rml::Event& event) override
- {
- using namespace Rml;
- switch (event.GetId())
- {
- case EventId::Keydown:
- {
- Rml::Input::KeyIdentifier key_identifier = (Rml::Input::KeyIdentifier)event.GetParameter<int>("key_identifier", 0);
- if (key_identifier == Rml::Input::KI_ESCAPE)
- Backend::RequestExit();
- }
- break;
- default: break;
- }
- }
- Rml::ElementDocument* GetDocument() { return document; }
- void SubmitForm(Rml::String in_submit_message)
- {
- submitting = true;
- submitting_start_time = Rml::GetSystemInterface()->GetElapsedTime();
- submit_message = in_submit_message;
- if (auto el_output = document->GetElementById("form_output"))
- el_output->SetInnerRML("");
- if (auto el_progress = document->GetElementById("submit_progress"))
- el_progress->SetProperty("display", "block");
- }
- void SetSandboxStylesheet(const Rml::String& string)
- {
- if (iframe && rml_basic_style_sheet)
- {
- auto style = Rml::MakeShared<Rml::StyleSheetContainer>();
- Rml::StreamMemory stream((const Rml::byte*)string.data(), string.size());
- stream.SetSourceURL("sandbox://rcss");
- style->LoadStyleSheetContainer(&stream);
- style = rml_basic_style_sheet->CombineStyleSheetContainer(*style);
- iframe->SetStyleSheetContainer(style);
- }
- }
- void SetSandboxBody(const Rml::String& string)
- {
- if (iframe)
- {
- iframe->SetInnerRML(string);
- }
- }
- private:
- Rml::ElementDocument* document = nullptr;
- Rml::ElementDocument* iframe = nullptr;
- Rml::Element *gauge = nullptr, *progress_horizontal = nullptr;
- Rml::SharedPtr<Rml::StyleSheetContainer> rml_basic_style_sheet;
- bool submitting = false;
- double submitting_start_time = 0;
- Rml::String submit_message;
- };
- Rml::UniquePtr<DemoWindow> demo_window;
- struct TweeningParameters {
- Rml::Tween::Type type = Rml::Tween::Linear;
- Rml::Tween::Direction direction = Rml::Tween::Out;
- float duration = 0.5f;
- } tweening_parameters;
- class DemoEventListener : public Rml::EventListener {
- public:
- DemoEventListener(const Rml::String& value, Rml::Element* element) : value(value), element(element) {}
- void ProcessEvent(Rml::Event& event) override
- {
- using namespace Rml;
- if (value == "exit")
- {
- // Test replacing the current element.
- // Need to be careful with regard to lifetime issues. The event's current element will be destroyed, so we cannot
- // use it after SetInnerRml(). The library should handle this case safely internally when propagating the event further.
- Element* parent = element->GetParentNode();
- parent->SetInnerRML("<button onclick='confirm_exit' onblur='cancel_exit' onmouseout='cancel_exit'>Are you sure?</button>");
- if (Element* child = parent->GetChild(0))
- child->Focus();
- }
- else if (value == "confirm_exit")
- {
- Backend::RequestExit();
- }
- else if (value == "cancel_exit")
- {
- if (Element* parent = element->GetParentNode())
- parent->SetInnerRML("<button id='exit' onclick='exit'>Exit</button>");
- }
- else if (value == "change_color")
- {
- Colourb color((byte)Math::RandomInteger(255), (byte)Math::RandomInteger(255), (byte)Math::RandomInteger(255));
- element->Animate("image-color", Property(color, Unit::COLOUR), tweening_parameters.duration,
- Tween(tweening_parameters.type, tweening_parameters.direction));
- event.StopPropagation();
- }
- else if (value == "move_child")
- {
- Vector2f mouse_pos(event.GetParameter("mouse_x", 0.0f), event.GetParameter("mouse_y", 0.0f));
- if (Element* child = element->GetFirstChild())
- {
- Vector2f new_pos =
- mouse_pos - element->GetAbsoluteOffset() - Vector2f(0.35f * child->GetClientWidth(), 0.9f * child->GetClientHeight());
- Property destination = Transform::MakeProperty({Transforms::Translate2D(new_pos.x, new_pos.y)});
- if (tweening_parameters.duration <= 0)
- child->SetProperty(PropertyId::Transform, destination);
- else
- child->Animate("transform", destination, tweening_parameters.duration,
- Tween(tweening_parameters.type, tweening_parameters.direction));
- }
- }
- else if (value == "tween_function")
- {
- static const SmallUnorderedMap<String, Tween::Type> tweening_functions = {{"back", Tween::Back}, {"bounce", Tween::Bounce},
- {"circular", Tween::Circular}, {"cubic", Tween::Cubic}, {"elastic", Tween::Elastic}, {"exponential", Tween::Exponential},
- {"linear", Tween::Linear}, {"quadratic", Tween::Quadratic}, {"quartic", Tween::Quartic}, {"quintic", Tween::Quintic},
- {"sine", Tween::Sine}};
- String value = event.GetParameter("value", String());
- auto it = tweening_functions.find(value);
- if (it != tweening_functions.end())
- tweening_parameters.type = it->second;
- else
- {
- RMLUI_ERROR;
- }
- }
- else if (value == "tween_direction")
- {
- String value = event.GetParameter("value", String());
- if (value == "in")
- tweening_parameters.direction = Tween::In;
- else if (value == "out")
- tweening_parameters.direction = Tween::Out;
- else if (value == "in-out")
- tweening_parameters.direction = Tween::InOut;
- else
- {
- RMLUI_ERROR;
- }
- }
- else if (value == "tween_duration")
- {
- float value = (float)std::atof(rmlui_static_cast<Rml::ElementFormControl*>(element)->GetValue().c_str());
- tweening_parameters.duration = value;
- if (auto el_duration = element->GetElementById("duration"))
- el_duration->SetInnerRML(CreateString(20, "%2.2f", value));
- }
- else if (value == "rating")
- {
- auto el_rating = element->GetElementById("rating");
- auto el_rating_emoji = element->GetElementById("rating_emoji");
- if (el_rating && el_rating_emoji)
- {
- enum { Sad, Mediocre, Exciting, Celebrate, Champion, CountEmojis };
- static const Rml::String emojis[CountEmojis] = {(const char*)u8"😢", (const char*)u8"😐", (const char*)u8"😮", (const char*)u8"😎",
- (const char*)u8"🏆"};
- int value = event.GetParameter("value", 50);
- Rml::String emoji;
- if (value <= 0)
- emoji = emojis[Sad];
- else if (value < 50)
- emoji = emojis[Mediocre];
- else if (value < 75)
- emoji = emojis[Exciting];
- else if (value < 100)
- emoji = emojis[Celebrate];
- else
- emoji = emojis[Champion];
- el_rating->SetInnerRML(Rml::CreateString(30, "%d%%", value));
- el_rating_emoji->SetInnerRML(emoji);
- }
- }
- else if (value == "submit_form")
- {
- const auto& p = event.GetParameters();
- Rml::String output = "<p>";
- for (auto& entry : p)
- {
- auto value = Rml::StringUtilities::EncodeRml(entry.second.Get<Rml::String>());
- if (entry.first == "message")
- value = "<br/>" + value;
- output += "<strong>" + entry.first + "</strong>: " + value + "<br/>";
- }
- output += "</p>";
- demo_window->SubmitForm(output);
- }
- else if (value == "set_sandbox_body")
- {
- if (auto source = rmlui_dynamic_cast<Rml::ElementFormControl*>(element->GetElementById("sandbox_rml_source")))
- {
- auto value = source->GetValue();
- demo_window->SetSandboxBody(value);
- }
- }
- else if (value == "set_sandbox_style")
- {
- if (auto source = rmlui_dynamic_cast<Rml::ElementFormControl*>(element->GetElementById("sandbox_rcss_source")))
- {
- auto value = source->GetValue();
- demo_window->SetSandboxStylesheet(value);
- }
- }
- }
- void OnDetach(Rml::Element* /*element*/) override { delete this; }
- private:
- Rml::String value;
- Rml::Element* element;
- };
- class DemoEventListenerInstancer : public Rml::EventListenerInstancer {
- public:
- Rml::EventListener* InstanceEventListener(const Rml::String& value, Rml::Element* element) override
- {
- return new DemoEventListener(value, element);
- }
- };
- #if defined RMLUI_PLATFORM_WIN32
- #include <RmlUi_Include_Windows.h>
- int APIENTRY WinMain(HINSTANCE /*instance_handle*/, HINSTANCE /*previous_instance_handle*/, char* /*command_line*/, int /*command_show*/)
- #else
- int main(int /*argc*/, char** /*argv*/)
- #endif
- {
- const int width = 1600;
- const int height = 890;
- // Initializes the shell which provides common functionality used by the included samples.
- if (!Shell::Initialize())
- return -1;
- // Constructs the system and render interfaces, creates a window, and attaches the renderer.
- if (!Backend::Initialize("Demo Sample", width, height, true))
- {
- Shell::Shutdown();
- return -1;
- }
- // Install the custom interfaces constructed by the backend before initializing RmlUi.
- Rml::SetSystemInterface(Backend::GetSystemInterface());
- Rml::SetRenderInterface(Backend::GetRenderInterface());
- // RmlUi initialisation.
- Rml::Initialise();
- // Create the main RmlUi context.
- Rml::Context* context = Rml::CreateContext("main", Rml::Vector2i(width, height));
- if (!context)
- {
- Rml::Shutdown();
- Backend::Shutdown();
- Shell::Shutdown();
- return -1;
- }
- Rml::Debugger::Initialise(context);
- DemoEventListenerInstancer event_listener_instancer;
- Rml::Factory::RegisterEventListenerInstancer(&event_listener_instancer);
- Shell::LoadFonts();
- demo_window = Rml::MakeUnique<DemoWindow>("Demo sample", context);
- demo_window->GetDocument()->AddEventListener(Rml::EventId::Keydown, demo_window.get());
- demo_window->GetDocument()->AddEventListener(Rml::EventId::Keyup, demo_window.get());
- demo_window->GetDocument()->AddEventListener(Rml::EventId::Animationend, demo_window.get());
- bool running = true;
- while (running)
- {
- demo_window->Update();
- running = Backend::ProcessEvents(context, &Shell::ProcessKeyDownShortcuts, true);
- context->Update();
- Backend::BeginFrame();
- context->Render();
- Backend::PresentFrame();
- }
- demo_window->Shutdown();
- // Shutdown RmlUi.
- Rml::Shutdown();
- Backend::Shutdown();
- Shell::Shutdown();
- demo_window.reset();
- return 0;
- }
|