#include "html.h"
#include "element.h"
#include "document.h"
#include "render_item.h"
#include "render_flex.h"
#include "render_inline.h"
#include "render_table.h"
#include "el_before_after.h"
namespace litehtml
{
#define LITEHTML_EMPTY_FUNC {}
#define LITEHTML_RETURN_FUNC(ret) {return ret;}
element::element(const document::ptr& doc) : m_doc(doc)
{
}
position element::get_placement() const
{
position pos;
bool is_first = true;
for(const auto& ri_el : m_renders)
{
auto ri = ri_el.lock();
if(ri)
{
position ri_pos = ri_el.lock()->get_placement();
if(is_first)
{
is_first = false;
pos = ri_pos;
} else
{
if(pos.x < ri_pos.x)
{
pos.x = ri_pos.x;
}
if(pos.y < ri_pos.y)
{
pos.y = ri_pos.y;
}
}
}
}
return pos;
}
bool element::is_inline() const
{
if( css().get_display() == display_inline ||
css().get_display() == display_inline_table ||
css().get_display() == display_inline_block ||
css().get_display() == display_inline_text ||
css().get_display() == display_inline_flex)
{
return true;
}
return false;
}
bool element::is_inline_box() const
{
if( css().get_display() == display_inline_table ||
css().get_display() == display_inline_block ||
css().get_display() == display_inline_flex)
{
return true;
}
return false;
}
bool element::is_ancestor(const ptr &el) const
{
element::ptr el_parent = parent();
while(el_parent && el_parent != el)
{
el_parent = el_parent->parent();
}
if(el_parent)
{
return true;
}
return false;
}
bool element::is_table_skip() const
{
return is_space() || is_comment() || css().get_display() == display_none;
}
string element::dump_get_name()
{
return "element";
}
std::vector> element::dump_get_attrs()
{
return m_css.dump_get_attrs();
}
void element::dump(dumper& cout)
{
cout.begin_node(dump_get_name());
auto attrs = dump_get_attrs();
if(!attrs.empty())
{
cout.begin_attrs_group("attributes");
for (const auto &attr: attrs)
{
cout.add_attr(std::get<0>(attr), std::get<1>(attr));
}
cout.end_attrs_group();
}
if(!m_children.empty())
{
cout.begin_attrs_group("children");
for (const auto &el: m_children)
{
el->dump(cout);
}
cout.end_attrs_group();
}
cout.end_node();
}
std::shared_ptr element::create_render_item(const std::shared_ptr& parent_ri)
{
std::shared_ptr ret;
if(css().get_display() == display_table_column ||
css().get_display() == display_table_column_group ||
css().get_display() == display_table_footer_group ||
css().get_display() == display_table_header_group ||
css().get_display() == display_table_row_group)
{
ret = std::make_shared(shared_from_this());
} else if(css().get_display() == display_table_row)
{
ret = std::make_shared(shared_from_this());
} else if(css().get_display() == display_block ||
css().get_display() == display_table_cell ||
css().get_display() == display_table_caption ||
css().get_display() == display_list_item ||
css().get_display() == display_inline_block)
{
ret = std::make_shared(shared_from_this());
} else if(css().get_display() == display_table || css().get_display() == display_inline_table)
{
ret = std::make_shared(shared_from_this());
} else if(css().get_display() == display_inline || css().get_display() == display_inline_text)
{
ret = std::make_shared(shared_from_this());
} else if(css().get_display() == display_flex || css().get_display() == display_inline_flex)
{
ret = std::make_shared(shared_from_this());
}
if(ret)
{
if (css().get_display() == display_table ||
css().get_display() == display_inline_table ||
css().get_display() == display_table_caption ||
css().get_display() == display_table_cell ||
css().get_display() == display_table_column ||
css().get_display() == display_table_column_group ||
css().get_display() == display_table_footer_group ||
css().get_display() == display_table_header_group ||
css().get_display() == display_table_row ||
css().get_display() == display_table_row_group)
{
get_document()->add_tabular(ret);
}
ret->parent(parent_ri);
for(const auto& el : m_children)
{
auto ri = el->create_render_item(ret);
if(ri)
{
ret->add_child(ri);
}
}
}
return ret;
}
bool element::requires_styles_update()
{
for (const auto& used_style : m_used_styles)
{
if(used_style->m_selector->is_media_valid())
{
int res = select(*(used_style->m_selector), true);
if( (res == select_no_match && used_style->m_used) || (res == select_match && !used_style->m_used) )
{
return true;
}
}
}
return false;
}
void element::add_render(const std::shared_ptr& ri)
{
m_renders.push_back(ri);
}
bool element::find_styles_changes( position::vector& redraw_boxes)
{
if(css().get_display() == display_inline_text)
{
return false;
}
bool ret = false;
if(requires_styles_update())
{
auto fetch_boxes = [&](const std::shared_ptr& el)
{
for(const auto& weak_ri : el->m_renders)
{
auto ri = weak_ri.lock();
if(ri)
{
position::vector boxes;
ri->get_rendering_boxes(boxes);
for (auto &box: boxes)
{
redraw_boxes.push_back(box);
}
}
}
};
fetch_boxes(shared_from_this());
for (auto& el : m_children)
{
fetch_boxes(el);
}
refresh_styles();
compute_styles();
ret = true;
}
for (auto& el : m_children)
{
if(el->find_styles_changes(redraw_boxes))
{
ret = true;
}
}
return ret;
}
element::ptr element::_add_before_after(int type, const style& /*style*/)
{
element::ptr el;
if(type == 0)
{
el = std::make_shared(get_document());
m_children.insert(m_children.begin(), el);
} else
{
el = std::make_shared(get_document());
m_children.insert(m_children.end(), el);
}
el->parent(shared_from_this());
return el;
}
bool element::is_block_formatting_context() const
{
if(m_css.get_display() == display_block)
{
auto par = parent();
if(par && (par->css().get_display() == display_inline_flex || par->css().get_display() == display_flex))
{
return true;
}
}
if( m_css.get_display() == display_inline_block ||
m_css.get_display() == display_table_cell ||
m_css.get_display() == display_inline_flex ||
m_css.get_display() == display_flex ||
m_css.get_display() == display_table_caption ||
is_root() ||
m_css.get_float() != float_none ||
m_css.get_position() == element_position_absolute ||
m_css.get_position() == element_position_fixed ||
m_css.get_overflow() > overflow_visible)
{
return true;
}
return false;
}
litehtml::string litehtml::element::get_counter_value(const string& counter_name)
{
std::map::iterator i;
if (find_counter(_id(counter_name), i))
{
return std::to_string(i->second);
}
return "0";
}
string litehtml::element::get_counters_value(const string_vector& parameters)
{
string result;
if (parameters.size() >= 2) {
const string_id counter_name_id = _id(parameters[0]);
string delims = parameters[1];
litehtml::trim(delims, "\"'");
string_vector values;
element::ptr current = shared_from_this();
while (current != nullptr)
{
auto map_iterator = current->m_counter_values.find(counter_name_id);
if (map_iterator != current->m_counter_values.end()) {
values.push_back(std::to_string(map_iterator->second));
}
current = current->parent();
}
if (values.empty()) {
// if no counter is found, instance one with value '0'
shared_from_this()->m_counter_values[counter_name_id] = 0;
result = "0";
}
else {
std::reverse(values.begin(), values.end());
litehtml::join_string(result, values, delims);
}
}
return result;
}
bool litehtml::element::find_counter(const string_id& counter_name_id, std::map::iterator& map_iterator) {
element::ptr current = shared_from_this();
while (current != nullptr)
{
map_iterator = current->m_counter_values.find(counter_name_id);
if (map_iterator != current->m_counter_values.end()) {
return true;
}
// on each level, search previous siblings too
std::vector siblings = current->get_siblings_before();
std::reverse(siblings.begin(), siblings.end());
for (const element::ptr& sibling : siblings) {
map_iterator = sibling->m_counter_values.find(counter_name_id);
if (map_iterator != sibling->m_counter_values.end()) {
return true;
}
}
current = current->parent();
}
return false;
}
std::vector litehtml::element::get_siblings_before() const
{
std::vector siblings;
if (parent() != nullptr) {
for (const element::ptr& sybling : parent()->children()) {
if (sybling == shared_from_this()) {
break;
}
siblings.push_back(sybling);
}
}
return siblings;
}
void litehtml::element::parse_counter_tokens(const string_vector& tokens, const int default_value, std::function handler) const {
int pos = 0;
while (pos < (int) tokens.size()) {
const string& name = tokens[pos];
int value = default_value;
if (pos < (int) tokens.size() - 1 && litehtml::is_number(tokens[pos + 1], false)) {
value = atoi(tokens[pos + 1].c_str());
pos += 2;
}
else {
pos += 1;
}
handler(_id(name), value);
}
}
void litehtml::element::increment_counter(const string_id& counter_name_id, const int increment)
{
std::map::iterator i;
if (find_counter(counter_name_id, i)) {
i->second = i->second + increment;
}
else {
// if counter is not found, initialize one on this element
m_counter_values[counter_name_id] = increment;
}
}
void litehtml::element::reset_counter(const string_id& counter_name_id, const int value)
{
m_counter_values[counter_name_id] = value;
}
const background* element::get_background(bool /*own_only*/) LITEHTML_RETURN_FUNC(nullptr)
void element::add_style( const style& /*style*/) LITEHTML_EMPTY_FUNC
void element::select_all(const css_selector& /*selector*/, elements_list& /*res*/) LITEHTML_EMPTY_FUNC
elements_list element::select_all(const css_selector& /*selector*/) LITEHTML_RETURN_FUNC(elements_list())
elements_list element::select_all(const string& /*selector*/) LITEHTML_RETURN_FUNC(elements_list())
element::ptr element::select_one( const css_selector& /*selector*/ ) LITEHTML_RETURN_FUNC(nullptr)
element::ptr element::select_one( const string& /*selector*/ ) LITEHTML_RETURN_FUNC(nullptr)
element::ptr element::find_adjacent_sibling(const element::ptr& /*el*/, const css_selector& /*selector*/, bool /*apply_pseudo*/ /*= true*/, bool* /*is_pseudo*/ /*= 0*/) LITEHTML_RETURN_FUNC(nullptr)
element::ptr element::find_sibling(const element::ptr& /*el*/, const css_selector& /*selector*/, bool /*apply_pseudo*/ /*= true*/, bool* /*is_pseudo*/ /*= 0*/) LITEHTML_RETURN_FUNC(nullptr)
bool element::is_nth_last_child(const element::ptr& /*el*/, int /*num*/, int /*off*/, bool /*of_type*/, const css_selector::vector& /*selector_list*/) const LITEHTML_RETURN_FUNC(false)
bool element::is_nth_child(const element::ptr&, int /*num*/, int /*off*/, bool /*of_type*/, const css_selector::vector& /*selector_list*/) const LITEHTML_RETURN_FUNC(false)
bool element::is_only_child(const element::ptr& /*el*/, bool /*of_type*/) const LITEHTML_RETURN_FUNC(false)
void element::get_content_size( size& /*sz*/, int /*max_width*/ ) LITEHTML_EMPTY_FUNC
bool element::appendChild(const ptr &/*el*/) LITEHTML_RETURN_FUNC(false)
bool element::removeChild(const ptr &/*el*/) LITEHTML_RETURN_FUNC(false)
void element::clearRecursive() LITEHTML_EMPTY_FUNC
string_id element::id() const LITEHTML_RETURN_FUNC(empty_id)
string_id element::tag() const LITEHTML_RETURN_FUNC(empty_id)
const char* element::get_tagName() const LITEHTML_RETURN_FUNC("")
void element::set_tagName( const char* /*tag*/ ) LITEHTML_EMPTY_FUNC
void element::set_data( const char* /*data*/ ) LITEHTML_EMPTY_FUNC
void element::set_attr( const char* /*name*/, const char* /*val*/ ) LITEHTML_EMPTY_FUNC
void element::apply_stylesheet( const litehtml::css& /*stylesheet*/ ) LITEHTML_EMPTY_FUNC
void element::refresh_styles() LITEHTML_EMPTY_FUNC
void element::on_click() LITEHTML_EMPTY_FUNC
void element::compute_styles( bool /*recursive*/ ) LITEHTML_EMPTY_FUNC
const char* element::get_attr( const char* /*name*/, const char* def /*= 0*/ ) const LITEHTML_RETURN_FUNC(def)
bool element::is_white_space() const LITEHTML_RETURN_FUNC(false)
bool element::is_space() const LITEHTML_RETURN_FUNC(false)
bool element::is_comment() const LITEHTML_RETURN_FUNC(false)
bool element::is_body() const LITEHTML_RETURN_FUNC(false)
bool element::is_break() const LITEHTML_RETURN_FUNC(false)
bool element::is_text() const LITEHTML_RETURN_FUNC(false)
bool element::on_mouse_over() LITEHTML_RETURN_FUNC(false)
bool element::on_mouse_leave() LITEHTML_RETURN_FUNC(false)
bool element::on_lbutton_down() LITEHTML_RETURN_FUNC(false)
bool element::on_lbutton_up() LITEHTML_RETURN_FUNC(false)
bool element::set_pseudo_class( string_id /*cls*/, bool /*add*/ ) LITEHTML_RETURN_FUNC(false)
bool element::set_class( const char* /*pclass*/, bool /*add*/ ) LITEHTML_RETURN_FUNC(false)
bool element::is_replaced() const LITEHTML_RETURN_FUNC(false)
void element::draw(uint_ptr /*hdc*/, int /*x*/, int /*y*/, const position */*clip*/, const std::shared_ptr &/*ri*/) LITEHTML_EMPTY_FUNC
void element::draw_background(uint_ptr /*hdc*/, int /*x*/, int /*y*/, const position */*clip*/, const std::shared_ptr &/*ri*/) LITEHTML_EMPTY_FUNC
void element::get_text( string& /*text*/ ) LITEHTML_EMPTY_FUNC
void element::parse_attributes() LITEHTML_EMPTY_FUNC
int element::select(const css_selector::vector& /*selector_list*/, bool /*apply_pseudo*/) LITEHTML_RETURN_FUNC(select_no_match)
int element::select(const string& /*selector*/) LITEHTML_RETURN_FUNC(select_no_match)
int element::select(const css_selector& /*selector*/, bool /*apply_pseudo*/) LITEHTML_RETURN_FUNC(select_no_match)
int element::select(const css_element_selector& /*selector*/, bool /*apply_pseudo*/) LITEHTML_RETURN_FUNC(select_no_match)
element::ptr element::find_ancestor(const css_selector& /*selector*/, bool /*apply_pseudo*/, bool* /*is_pseudo*/) LITEHTML_RETURN_FUNC(nullptr)
} // namespace litehtml