2022-10-20 23:30:39 +03:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2022, kleines Filmröllchen <filmroellchen@serenityos.org>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "SlideObject.h"
|
2023-01-09 02:44:32 +03:00
|
|
|
#include "Presentation.h"
|
2022-10-20 23:30:39 +03:00
|
|
|
#include <AK/JsonObject.h>
|
2023-01-09 02:44:32 +03:00
|
|
|
#include <AK/URL.h>
|
|
|
|
#include <LibGfx/Font/FontStyleMapping.h>
|
|
|
|
#include <LibGfx/Rect.h>
|
2022-10-20 23:30:39 +03:00
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
static DeprecatedString to_css_length(float design_value, Presentation const& presentation)
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
float length_in_vw = design_value / static_cast<float>(presentation.normative_size().width()) * 100.0f;
|
|
|
|
return DeprecatedString::formatted("{}vw", length_in_vw);
|
|
|
|
}
|
2022-10-20 23:30:39 +03:00
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
ErrorOr<NonnullRefPtr<SlideObject>> SlideObject::parse_slide_object(JsonObject const& slide_object_json)
|
|
|
|
{
|
2022-12-21 14:42:06 +03:00
|
|
|
auto const& maybe_type = slide_object_json.get_deprecated("type"sv);
|
2022-10-20 23:30:39 +03:00
|
|
|
if (!maybe_type.is_string())
|
|
|
|
return Error::from_string_view("Slide object must have a type"sv);
|
|
|
|
|
|
|
|
auto type = maybe_type.as_string();
|
|
|
|
RefPtr<SlideObject> object;
|
|
|
|
if (type == "text"sv)
|
|
|
|
object = TRY(try_make_ref_counted<Text>());
|
|
|
|
else if (type == "image"sv)
|
2023-01-09 02:44:32 +03:00
|
|
|
object = TRY(try_make_ref_counted<Image>());
|
2022-10-20 23:30:39 +03:00
|
|
|
else
|
|
|
|
return Error::from_string_view("Unsupported slide object type"sv);
|
|
|
|
|
|
|
|
slide_object_json.for_each_member([&](auto const& key, auto const& value) {
|
2023-01-09 02:44:32 +03:00
|
|
|
object->set_property(key, value);
|
2022-10-20 23:30:39 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
return object.release_nonnull();
|
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
void SlideObject::set_property(StringView name, JsonValue value)
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
if (name == "rect"sv) {
|
|
|
|
if (value.is_array() && value.as_array().size() == 4) {
|
|
|
|
Gfx::IntRect rect;
|
|
|
|
rect.set_x(value.as_array()[0].to_i32());
|
|
|
|
rect.set_y(value.as_array()[1].to_i32());
|
|
|
|
rect.set_width(value.as_array()[2].to_i32());
|
|
|
|
rect.set_height(value.as_array()[3].to_i32());
|
|
|
|
m_rect = rect;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_properties.set(name, move(value));
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
void GraphicsObject::set_property(StringView name, JsonValue value)
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
if (name == "color"sv) {
|
|
|
|
if (auto color = Gfx::Color::from_string(value.to_deprecated_string()); color.has_value()) {
|
|
|
|
m_color = color.release_value();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
SlideObject::set_property(name, move(value));
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
void Text::set_property(StringView name, JsonValue value)
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
if (name == "text"sv) {
|
|
|
|
m_text = value.to_deprecated_string();
|
|
|
|
} else if (name == "font"sv) {
|
|
|
|
m_font_family = value.to_deprecated_string();
|
|
|
|
} else if (name == "font-weight"sv) {
|
|
|
|
m_font_weight = Gfx::name_to_weight(value.to_deprecated_string());
|
|
|
|
} else if (name == "font-size"sv) {
|
|
|
|
m_font_size_in_pt = value.to_float();
|
|
|
|
} else if (name == "text-alignment"sv) {
|
|
|
|
m_text_align = value.to_deprecated_string();
|
|
|
|
}
|
|
|
|
GraphicsObject::set_property(name, move(value));
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
void Image::set_property(StringView name, JsonValue value)
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
if (name == "path"sv) {
|
|
|
|
m_src = value.to_deprecated_string();
|
|
|
|
} else if (name == "scaling-mode"sv) {
|
|
|
|
if (value.to_deprecated_string() == "nearest-neighbor"sv)
|
|
|
|
m_image_rendering = "crisp-edges"sv;
|
|
|
|
else if (value.to_deprecated_string() == "smooth-pixels"sv)
|
|
|
|
m_image_rendering = "pixelated"sv;
|
|
|
|
}
|
|
|
|
SlideObject::set_property(name, move(value));
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
ErrorOr<HTMLElement> Text::render(Presentation const& presentation) const
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
HTMLElement div;
|
|
|
|
div.tag_name = "div"sv;
|
|
|
|
div.style.set("color"sv, m_color.to_deprecated_string());
|
|
|
|
div.style.set("font-family"sv, DeprecatedString::formatted("'{}'", m_font_family));
|
|
|
|
div.style.set("font-size"sv, to_css_length(m_font_size_in_pt * 1.33333333f, presentation));
|
|
|
|
div.style.set("font-weight"sv, DeprecatedString::number(m_font_weight));
|
|
|
|
div.style.set("text-align"sv, m_text_align);
|
|
|
|
div.style.set("white-space"sv, "pre-wrap"sv);
|
|
|
|
div.style.set("width"sv, to_css_length(m_rect.width(), presentation));
|
|
|
|
div.style.set("height"sv, to_css_length(m_rect.height(), presentation));
|
|
|
|
div.style.set("position"sv, "absolute"sv);
|
|
|
|
div.style.set("left"sv, to_css_length(m_rect.left(), presentation));
|
|
|
|
div.style.set("top"sv, to_css_length(m_rect.top(), presentation));
|
|
|
|
div.inner_text = m_text;
|
|
|
|
return div;
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
ErrorOr<HTMLElement> Image::render(Presentation const& presentation) const
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
HTMLElement img;
|
|
|
|
img.tag_name = "img"sv;
|
|
|
|
img.attributes.set("src"sv, URL::create_with_file_scheme(m_src).to_deprecated_string());
|
|
|
|
img.style.set("image-rendering"sv, m_image_rendering);
|
|
|
|
if (m_rect.width() > m_rect.height())
|
|
|
|
img.style.set("height"sv, "100%"sv);
|
|
|
|
else
|
|
|
|
img.style.set("width"sv, "100%"sv);
|
|
|
|
|
|
|
|
HTMLElement image_wrapper;
|
|
|
|
image_wrapper.tag_name = "div"sv;
|
|
|
|
image_wrapper.children.append(move(img));
|
|
|
|
image_wrapper.style.set("position"sv, "absolute"sv);
|
|
|
|
image_wrapper.style.set("left"sv, to_css_length(m_rect.left(), presentation));
|
|
|
|
image_wrapper.style.set("top"sv, to_css_length(m_rect.top(), presentation));
|
|
|
|
image_wrapper.style.set("width"sv, to_css_length(m_rect.width(), presentation));
|
|
|
|
image_wrapper.style.set("height"sv, to_css_length(m_rect.height(), presentation));
|
|
|
|
image_wrapper.style.set("text-align"sv, "center"sv);
|
|
|
|
return image_wrapper;
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
|
|
|
|
2023-01-09 02:44:32 +03:00
|
|
|
ErrorOr<void> HTMLElement::serialize(StringBuilder& builder) const
|
2022-10-20 23:30:39 +03:00
|
|
|
{
|
2023-01-09 02:44:32 +03:00
|
|
|
TRY(builder.try_appendff("<{}", tag_name));
|
|
|
|
for (auto const& [key, value] : attributes) {
|
|
|
|
// FIXME: Escape the value string as necessary.
|
|
|
|
TRY(builder.try_appendff(" {}='{}'", key, value));
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|
2023-01-09 02:44:32 +03:00
|
|
|
TRY(builder.try_append(" style=\""sv));
|
|
|
|
for (auto const& [key, value] : style) {
|
|
|
|
// FIXME: Escape the value string as necessary.
|
|
|
|
TRY(builder.try_appendff(" {}: {};", key, value));
|
|
|
|
}
|
|
|
|
TRY(builder.try_append("\">"sv));
|
|
|
|
if (!inner_text.is_empty())
|
|
|
|
TRY(builder.try_append(inner_text));
|
|
|
|
for (auto const& child : children) {
|
|
|
|
TRY(child.serialize(builder));
|
|
|
|
}
|
|
|
|
TRY(builder.try_appendff("</{}>", tag_name));
|
|
|
|
return {};
|
2022-10-20 23:30:39 +03:00
|
|
|
}
|