2018-10-03 22:02:46 +08:00
|
|
|
|
// Copyright (c) 2016-2018 Easy2D - Nomango
|
|
|
|
|
|
//
|
|
|
|
|
|
// 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.
|
|
|
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
|
#include "Node.h"
|
|
|
|
|
|
#include "Scene.h"
|
|
|
|
|
|
#include "Task.h"
|
2018-11-14 01:34:41 +08:00
|
|
|
|
#include "Action.hpp"
|
2018-11-08 21:39:26 +08:00
|
|
|
|
#include "render.h"
|
2018-11-15 17:59:18 +08:00
|
|
|
|
#include "logs.h"
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
namespace easy2d
|
|
|
|
|
|
{
|
2018-11-15 14:35:19 +08:00
|
|
|
|
namespace
|
|
|
|
|
|
{
|
|
|
|
|
|
float default_pivot_x = 0.f;
|
|
|
|
|
|
float default_pivot_y = 0.f;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Node::SetDefaultPivot(float pivot_x, float pivot_y)
|
|
|
|
|
|
{
|
|
|
|
|
|
default_pivot_x = pivot_x;
|
|
|
|
|
|
default_pivot_y = pivot_y;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
Node::Node()
|
|
|
|
|
|
: visible_(true)
|
|
|
|
|
|
, parent_(nullptr)
|
|
|
|
|
|
, hash_name_(0)
|
|
|
|
|
|
, dirty_sort_(false)
|
|
|
|
|
|
, order_(0)
|
2018-11-18 20:26:41 +08:00
|
|
|
|
, opacity_(1.f)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
, display_opacity_(1.f)
|
|
|
|
|
|
, children_()
|
2018-11-20 01:20:06 +08:00
|
|
|
|
, border_color_(Color::Red, 0.6f)
|
2018-11-12 02:10:35 +08:00
|
|
|
|
, initial_matrix_()
|
|
|
|
|
|
, final_matrix_()
|
2018-11-20 01:20:06 +08:00
|
|
|
|
, pivot_(default_pivot_x, default_pivot_y)
|
|
|
|
|
|
, size_()
|
2018-09-05 00:08:03 +08:00
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
Node::~Node()
|
2018-09-07 23:47:21 +08:00
|
|
|
|
{
|
2018-07-28 18:44:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::Visit()
|
2018-06-01 17:53:56 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (!visible_)
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
2018-11-18 20:26:41 +08:00
|
|
|
|
UpdateTransform();
|
|
|
|
|
|
|
2018-11-18 23:41:15 +08:00
|
|
|
|
auto graphics = devices::Graphics::Instance();
|
|
|
|
|
|
|
2018-11-15 23:40:13 +08:00
|
|
|
|
if (children_.IsEmpty())
|
2018-01-30 16:45:38 +08:00
|
|
|
|
{
|
2018-11-15 14:35:19 +08:00
|
|
|
|
graphics->SetTransform(final_matrix_);
|
2018-11-18 20:26:41 +08:00
|
|
|
|
graphics->SetOpacity(display_opacity_);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
OnDraw();
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
if (dirty_sort_)
|
2018-01-30 16:45:38 +08:00
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
children_.Sort(
|
2018-11-14 01:34:41 +08:00
|
|
|
|
[](spNode const& n1, spNode const& n2) { return n1->GetOrder() < n2->GetOrder(); }
|
2018-11-08 00:21:59 +08:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
dirty_sort_ = false;
|
2018-01-30 16:45:38 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
2018-11-15 23:40:13 +08:00
|
|
|
|
spNode child = children_.First();
|
|
|
|
|
|
for (spNode next; child; child = next)
|
2018-01-30 16:45:38 +08:00
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
next = child->NextItem();
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (child->GetOrder() < 0)
|
|
|
|
|
|
{
|
|
|
|
|
|
child->Visit();
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
2018-01-30 16:45:38 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
2018-11-15 14:35:19 +08:00
|
|
|
|
graphics->SetTransform(final_matrix_);
|
2018-11-18 20:26:41 +08:00
|
|
|
|
graphics->SetOpacity(display_opacity_);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
OnDraw();
|
|
|
|
|
|
|
2018-11-15 23:40:13 +08:00
|
|
|
|
for (spNode next; child; child = next)
|
|
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
next = child->NextItem();
|
2018-11-15 23:40:13 +08:00
|
|
|
|
child->Visit();
|
|
|
|
|
|
}
|
2018-01-30 16:45:38 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2018-07-28 18:44:37 +08:00
|
|
|
|
|
2018-11-14 16:39:24 +08:00
|
|
|
|
void Node::Update(Duration const& dt)
|
2018-07-28 18:44:37 +08:00
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
if (children_.IsEmpty())
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
|
|
|
|
|
OnUpdate(dt);
|
2018-11-14 16:39:24 +08:00
|
|
|
|
UpdateActions(this, dt);
|
|
|
|
|
|
UpdateTasks(dt);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
// <20><><EFBFBD><EFBFBD> Order С<><D0A1><EFBFBD><EFBFBD><EFBFBD>Ľڵ<C4BD>
|
|
|
|
|
|
spNode child = children_.First();
|
|
|
|
|
|
for (spNode next; child; child = next)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
|
|
|
|
|
if (child->GetOrder() < 0)
|
|
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
next = child->NextItem();
|
2018-11-14 16:39:24 +08:00
|
|
|
|
child->Update(dt);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
OnUpdate(dt);
|
2018-11-14 16:39:24 +08:00
|
|
|
|
UpdateActions(this, dt);
|
|
|
|
|
|
UpdateTasks(dt);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
2018-11-15 23:40:13 +08:00
|
|
|
|
for (spNode next; child; child = next)
|
|
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
next = child->NextItem();
|
2018-11-15 23:40:13 +08:00
|
|
|
|
child->Update(dt);
|
|
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2018-10-06 09:45:28 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
|
|
|
|
|
void Node::DrawBorder()
|
2018-10-06 09:45:28 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (visible_)
|
2018-10-06 09:45:28 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (border_)
|
2018-10-06 09:45:28 +08:00
|
|
|
|
{
|
2018-11-18 20:26:41 +08:00
|
|
|
|
devices::Graphics::Instance()->DrawGeometry(border_, border_color_, 1.5f);
|
2018-10-06 09:45:28 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
DrawChildrenBorder();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Node::DrawChildrenBorder()
|
|
|
|
|
|
{
|
|
|
|
|
|
for (auto child = children_.First(); child; child = child->NextItem())
|
|
|
|
|
|
{
|
|
|
|
|
|
child->DrawBorder();
|
2018-10-06 09:45:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
math::Matrix const & Node::GetTransformMatrix()
|
|
|
|
|
|
{
|
|
|
|
|
|
UpdateTransform();
|
|
|
|
|
|
return final_matrix_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::UpdateTransform()
|
2018-07-07 18:04:18 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (!dirty_transform_)
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
dirty_transform_ = false;
|
|
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
Point center{ size_.width * pivot_.x, size_.height * pivot_.y };
|
|
|
|
|
|
final_matrix_ = math::Matrix::Scaling(transform_.scale, center)
|
|
|
|
|
|
* math::Matrix::Skewing(transform_.skew.x, transform_.skew.y, center)
|
|
|
|
|
|
* math::Matrix::Rotation(transform_.rotation, center)
|
|
|
|
|
|
* math::Matrix::Translation(transform_.position - center);
|
|
|
|
|
|
|
2018-11-15 01:02:05 +08:00
|
|
|
|
initial_matrix_ = final_matrix_ * math::Matrix::Translation(
|
2018-11-20 01:20:06 +08:00
|
|
|
|
Point{ size_.width * pivot_.x, size_.height * pivot_.y }
|
2018-11-08 00:21:59 +08:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
if (parent_)
|
2018-09-01 23:00:08 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
initial_matrix_ = initial_matrix_ * parent_->initial_matrix_;
|
|
|
|
|
|
final_matrix_ = final_matrix_ * parent_->initial_matrix_;
|
|
|
|
|
|
}
|
2018-07-07 18:04:18 +08:00
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
UpdateBorder();
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
2018-11-17 17:15:32 +08:00
|
|
|
|
for (auto child = children_.First(); child; child = child->NextItem())
|
2018-07-28 18:44:37 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
child->dirty_transform_ = true;
|
2018-07-28 18:44:37 +08:00
|
|
|
|
}
|
2018-07-07 18:04:18 +08:00
|
|
|
|
}
|
2018-06-01 17:53:56 +08:00
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
void Node::UpdateBorder()
|
|
|
|
|
|
{
|
|
|
|
|
|
cpRectangleGeometry rect;
|
|
|
|
|
|
cpTransformedGeometry transformed;
|
|
|
|
|
|
|
|
|
|
|
|
ThrowIfFailed(
|
|
|
|
|
|
devices::Graphics::Instance()->CreateRectangleGeometry(
|
|
|
|
|
|
rect,
|
|
|
|
|
|
Rect(Point{}, size_)
|
|
|
|
|
|
)
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
ThrowIfFailed(
|
|
|
|
|
|
devices::Graphics::Instance()->CreateTransformedGeometry(
|
|
|
|
|
|
transformed,
|
|
|
|
|
|
final_matrix_,
|
|
|
|
|
|
rect
|
|
|
|
|
|
)
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
border_ = transformed;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
bool Node::Dispatch(const MouseEvent & e, bool handled)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (visible_)
|
|
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
spNode prev;
|
2018-11-17 17:15:32 +08:00
|
|
|
|
for (auto child = children_.Last(); child; child = prev)
|
2018-11-15 23:40:13 +08:00
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
prev = child->PrevItem();
|
2018-11-15 23:40:13 +08:00
|
|
|
|
handled = child->Dispatch(e, handled);
|
|
|
|
|
|
}
|
2018-07-13 00:45:39 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
auto handler = dynamic_cast<MouseEventHandler*>(this);
|
|
|
|
|
|
if (handler)
|
|
|
|
|
|
handler->Handle(e);
|
|
|
|
|
|
}
|
2018-09-16 16:07:51 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return handled;
|
|
|
|
|
|
}
|
2018-09-11 00:37:52 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
bool Node::Dispatch(const KeyEvent & e, bool handled)
|
2018-09-11 00:37:52 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (visible_)
|
|
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
spNode prev;
|
2018-11-17 17:15:32 +08:00
|
|
|
|
for (auto child = children_.Last(); child; child = prev)
|
2018-11-15 23:40:13 +08:00
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
prev = child->PrevItem();
|
2018-11-15 23:40:13 +08:00
|
|
|
|
handled = child->Dispatch(e, handled);
|
|
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
|
|
|
|
|
auto handler = dynamic_cast<KeyEventHandler*>(this);
|
|
|
|
|
|
if (handler)
|
|
|
|
|
|
handler->Handle(e);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return handled;
|
2018-09-11 00:37:52 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
|
|
|
|
|
|
void Node::UpdateOpacity()
|
2018-07-13 00:45:39 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (parent_)
|
|
|
|
|
|
{
|
2018-11-18 20:26:41 +08:00
|
|
|
|
display_opacity_ = opacity_ * parent_->display_opacity_;
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2018-11-17 17:15:32 +08:00
|
|
|
|
for (auto child = children_.First(); child; child = child->NextItem())
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
|
|
|
|
|
child->UpdateOpacity();
|
|
|
|
|
|
}
|
2018-07-13 00:45:39 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetOrder(int order)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (order_ == order)
|
|
|
|
|
|
return;
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
order_ = order;
|
|
|
|
|
|
if (parent_)
|
|
|
|
|
|
{
|
|
|
|
|
|
parent_->dirty_sort_ = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetOpacity(float opacity)
|
|
|
|
|
|
{
|
2018-11-18 20:26:41 +08:00
|
|
|
|
if (opacity_ == opacity)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return;
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-18 20:26:41 +08:00
|
|
|
|
display_opacity_ = opacity_ = std::min(std::max(opacity, 0.f), 1.f);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
UpdateOpacity();
|
|
|
|
|
|
}
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetPivotX(float pivot_x)
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
this->SetPivot(pivot_x, pivot_.y);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetPivotY(float pivot_y)
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
this->SetPivot(pivot_.x, pivot_y);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2017-10-15 02:46:24 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetPivot(float pivot_x, float pivot_y)
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
if (pivot_.x == pivot_x && pivot_.y == pivot_y)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return;
|
2017-10-15 02:46:24 +08:00
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
pivot_.x = pivot_x;
|
|
|
|
|
|
pivot_.y = pivot_y;
|
2018-11-08 00:21:59 +08:00
|
|
|
|
dirty_transform_ = true;
|
|
|
|
|
|
}
|
2017-10-15 02:46:24 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetWidth(float width)
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
this->SetSize(width, size_.height);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2017-10-15 02:46:24 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetHeight(float height)
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
this->SetSize(size_.width, height);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2017-10-15 23:58:39 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::SetSize(const Size& size)
|
|
|
|
|
|
{
|
|
|
|
|
|
this->SetSize(size.width, size.height);
|
|
|
|
|
|
}
|
2017-10-15 23:58:39 +08:00
|
|
|
|
|
2018-11-20 01:20:06 +08:00
|
|
|
|
void Node::SetSize(float width, float height)
|
2018-11-14 16:39:24 +08:00
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
if (size_.width == width && size_.height == height)
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
size_.width = width;
|
|
|
|
|
|
size_.height = height;
|
|
|
|
|
|
dirty_transform_ = true;
|
2018-11-14 16:39:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-16 17:19:03 +08:00
|
|
|
|
void Node::SetTransform(Transform const& transform)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
|
|
|
|
|
transform_ = transform;
|
|
|
|
|
|
dirty_transform_ = true;
|
|
|
|
|
|
}
|
2017-10-15 23:58:39 +08:00
|
|
|
|
|
2018-11-17 17:15:32 +08:00
|
|
|
|
void Node::SetBorderColor(Color const& color)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
|
|
|
|
|
border_color_ = color;
|
|
|
|
|
|
}
|
2018-03-01 19:28:22 +08:00
|
|
|
|
|
2018-11-14 01:34:41 +08:00
|
|
|
|
void Node::AddChild(spNode const& child, int order)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
2018-11-15 17:59:18 +08:00
|
|
|
|
if (!child)
|
|
|
|
|
|
logs::Warningln("Node::AddChild failed, child is nullptr");
|
2018-03-01 19:28:22 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (child)
|
|
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
#ifdef E2D_DEBUG
|
2018-11-14 01:34:41 +08:00
|
|
|
|
if (child->parent_)
|
2018-11-16 15:53:39 +08:00
|
|
|
|
logs::Errorln("The node to be added already has a parent");
|
|
|
|
|
|
for (Node* parent = parent_; parent; parent = parent->parent_)
|
|
|
|
|
|
if (parent == child)
|
|
|
|
|
|
logs::Errorln("A node cannot be its own parent");
|
|
|
|
|
|
#endif // E2D_DEBUG
|
2018-03-01 19:28:22 +08:00
|
|
|
|
|
2018-11-16 15:53:39 +08:00
|
|
|
|
children_.PushBack(Node::ItemType(child));
|
2018-11-08 00:21:59 +08:00
|
|
|
|
child->parent_ = this;
|
|
|
|
|
|
child->dirty_transform_ = true;
|
2018-11-16 15:53:39 +08:00
|
|
|
|
child->SetOrder(order);
|
|
|
|
|
|
child->UpdateOpacity();
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
dirty_sort_ = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2018-04-21 18:42:07 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::AddChild(const Nodes& nodes, int order)
|
|
|
|
|
|
{
|
|
|
|
|
|
for (const auto& node : nodes)
|
|
|
|
|
|
{
|
|
|
|
|
|
this->AddChild(node, order);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2018-07-28 18:44:37 +08:00
|
|
|
|
|
2018-11-16 16:16:16 +08:00
|
|
|
|
Rect Node::GetBounds()
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
return Rect(Point{}, size_);
|
2018-11-16 16:16:16 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-17 17:15:32 +08:00
|
|
|
|
Node::Nodes Node::GetChildren(String const& name) const
|
2017-10-14 11:40:47 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
Nodes children;
|
2018-11-08 21:39:26 +08:00
|
|
|
|
size_t hash_code = std::hash<String>{}(name);
|
2018-02-03 22:04:43 +08:00
|
|
|
|
|
2018-11-16 15:53:39 +08:00
|
|
|
|
for (auto child = children_.First(); child != children_.Last(); child = child->NextItem())
|
2017-10-14 11:40:47 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (child->hash_name_ == hash_code && child->name_ == name)
|
2018-05-24 00:58:16 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
children.push_back(child);
|
2018-05-24 00:58:16 +08:00
|
|
|
|
}
|
2017-10-14 11:40:47 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return children;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-17 17:15:32 +08:00
|
|
|
|
spNode Node::GetChild(String const& name) const
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
2018-11-08 21:39:26 +08:00
|
|
|
|
size_t hash_code = std::hash<String>{}(name);
|
2017-10-14 18:43:32 +08:00
|
|
|
|
|
2018-11-16 15:53:39 +08:00
|
|
|
|
for (auto child = children_.First(); child != children_.Last(); child = child->NextItem())
|
2017-10-17 21:22:25 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (child->hash_name_ == hash_code && child->name_ == name)
|
|
|
|
|
|
{
|
|
|
|
|
|
return child;
|
|
|
|
|
|
}
|
2017-10-17 21:22:25 +08:00
|
|
|
|
}
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return nullptr;
|
2017-10-14 11:40:47 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-15 23:40:13 +08:00
|
|
|
|
Node::Children const & Node::GetChildren() const
|
2018-03-31 18:12:01 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return children_;
|
2018-03-31 18:12:01 +08:00
|
|
|
|
}
|
2017-10-12 23:34:13 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::RemoveFromParent()
|
2018-04-24 13:28:21 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (parent_)
|
2018-04-24 13:28:21 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
parent_->RemoveChild(this);
|
2018-04-24 13:28:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-14 01:34:41 +08:00
|
|
|
|
bool Node::RemoveChild(spNode const& child)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
2018-11-15 17:59:18 +08:00
|
|
|
|
if (!child)
|
|
|
|
|
|
logs::Warningln("Node::RemoveChild failed, child is nullptr");
|
2018-02-03 22:04:43 +08:00
|
|
|
|
|
2018-11-15 23:40:13 +08:00
|
|
|
|
if (children_.IsEmpty())
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
2018-02-03 22:04:43 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (child)
|
|
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
children_.Remove(Node::ItemType(child));
|
2018-11-15 23:40:13 +08:00
|
|
|
|
return true;
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2018-08-23 16:37:51 +08:00
|
|
|
|
return false;
|
2017-10-14 18:43:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-17 17:15:32 +08:00
|
|
|
|
void Node::RemoveChildren(String const& child_name)
|
2017-10-14 18:43:32 +08:00
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
if (children_.IsEmpty())
|
2017-10-14 18:43:32 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
2018-02-03 22:04:43 +08:00
|
|
|
|
|
2018-11-08 21:39:26 +08:00
|
|
|
|
size_t hash_code = std::hash<String>{}(child_name);
|
2018-11-15 23:40:13 +08:00
|
|
|
|
spNode next;
|
2018-11-17 17:15:32 +08:00
|
|
|
|
for (auto child = children_.First(); child; child = next)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
{
|
2018-11-16 15:53:39 +08:00
|
|
|
|
next = child->NextItem();
|
2018-11-15 23:40:13 +08:00
|
|
|
|
|
|
|
|
|
|
if (child->hash_name_ == hash_code && child->name_ == child_name)
|
|
|
|
|
|
children_.Remove(child);
|
2017-10-14 18:43:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-10-14 11:40:47 +08:00
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
void Node::RemoveAllChildren()
|
2017-10-14 18:43:32 +08:00
|
|
|
|
{
|
2018-11-15 23:40:13 +08:00
|
|
|
|
children_.Clear();
|
2017-10-14 18:43:32 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-08 00:21:59 +08:00
|
|
|
|
bool Node::ContainsPoint(const Point& point)
|
|
|
|
|
|
{
|
2018-11-20 01:20:06 +08:00
|
|
|
|
if (size_.width == 0.f || size_.height == 0.f)
|
2018-11-08 00:21:59 +08:00
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
|
|
UpdateTransform();
|
|
|
|
|
|
|
|
|
|
|
|
BOOL ret = 0;
|
|
|
|
|
|
ThrowIfFailed(
|
|
|
|
|
|
border_->FillContainsPoint(
|
2018-11-20 01:20:06 +08:00
|
|
|
|
point,
|
2018-11-08 00:21:59 +08:00
|
|
|
|
D2D1::Matrix3x2F::Identity(),
|
|
|
|
|
|
&ret
|
|
|
|
|
|
)
|
|
|
|
|
|
);
|
2018-11-20 01:20:06 +08:00
|
|
|
|
return !!ret;
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2018-11-11 16:32:12 +08:00
|
|
|
|
void Node::SetVisible(bool val)
|
2018-09-05 00:08:03 +08:00
|
|
|
|
{
|
2018-11-11 16:32:12 +08:00
|
|
|
|
visible_ = val;
|
2018-09-05 00:08:03 +08:00
|
|
|
|
}
|
2017-10-21 19:09:31 +08:00
|
|
|
|
|
2018-11-17 17:15:32 +08:00
|
|
|
|
void Node::SetName(String const& name)
|
2017-10-14 11:40:47 +08:00
|
|
|
|
{
|
2018-11-08 00:21:59 +08:00
|
|
|
|
if (name_ != name)
|
|
|
|
|
|
{
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD><EFBFBD>ڵ<EFBFBD><DAB5><EFBFBD>
|
|
|
|
|
|
name_ = name;
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD><EFBFBD>ڵ<EFBFBD> Hash <20><>
|
2018-11-08 21:39:26 +08:00
|
|
|
|
hash_name_ = std::hash<String>{}(name);
|
2018-11-08 00:21:59 +08:00
|
|
|
|
}
|
2017-10-14 11:40:47 +08:00
|
|
|
|
}
|
2017-10-17 21:22:25 +08:00
|
|
|
|
}
|