// Ryzom - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "nel/gui/ctrl_polygon.h"
#include "nel/gui/widget_manager.h"
#include "nel/gui/view_renderer.h"
#include "nel/gui/interface_group.h"
using namespace NLMISC;
namespace NLGUI
{
// *********************************************************************************
CCtrlPolygon::CCtrlPolygon( const TCtorParam ¶m ) : CCtrlBase( param )
{
// Construct
_Color = CRGBA::White;
//_Matrix = CMatrix::Identity;
_Valid = true;
}
// *********************************************************************************
void CCtrlPolygon::updateBoudingRect()
{
H_AUTO(Rz_CCtrlPolygon_updateBoudingRect)
if (_Poly.Vertices.empty())
{
setX(0);
setY(0);
setW(0);
setH(0);
return;
}
//
sint32 xmin = INT_MAX;
sint32 ymin = INT_MAX;
sint32 xmax = INT_MIN;
sint32 ymax = INT_MIN;
uint numVerts = (uint)_Poly.Vertices.size();
_XFormPoly.Vertices.resize(numVerts);
for(uint k = 0; k < numVerts; ++k)
{
CVector2f &finalPos = _XFormPoly.Vertices[k];
//finalPos = _Matrix * _Poly.Vertices[k];
computeScaledVertex(finalPos, CVector2f(_Poly.Vertices[k].x, _Poly.Vertices[k].y));
xmin = std::min(xmin, (sint32) floorf(finalPos.x));
xmax = std::max(xmax, (sint32) ceilf(finalPos.x));
ymin = std::min(ymin, (sint32) floorf(finalPos.y));
ymax = std::max(ymax, (sint32) ceilf(finalPos.y));
}
setX(xmin);
setY(ymin);
setW(xmax - xmin);
setH(ymax - ymin);
}
// *********************************************************************************
bool CCtrlPolygon::contains(const CVector2f &pos) const
{
H_AUTO(Rz_CCtrlPolygon_contains)
if (!_Valid) return false;
return _XFormPoly.contains(pos, false);
}
// *********************************************************************************
void CCtrlPolygon::setVertices(const std::vector &vertices)
{
H_AUTO(Rz_CCtrlPolygon_setVertices)
if (vertices.size() == _Poly.Vertices.size() &&
std::equal(vertices.begin(), vertices.end(), _Poly.Vertices.begin())) return; // remains unchanged
//TTicks startTime = CTime::getPerformanceTime();
_Poly.Vertices = vertices;
_Tris.clear();
std::list polys;
bool splitDone = _Poly.toConvexPolygons(polys, NLMISC::CMatrix::Identity);
if (!splitDone)
{
polys.clear();
// maybe wrong orientation
std::reverse(_Poly.Vertices.begin(), _Poly.Vertices.end());
splitDone = _Poly.toConvexPolygons(polys, NLMISC::CMatrix::Identity);
std::reverse(_Poly.Vertices.begin(), _Poly.Vertices.end());
}
_Tris.clear();
if (splitDone)
{
for(std::list::iterator it = polys.begin(); it != polys.end(); ++it)
{
it->toTriFan(_Tris);
}
}
_Touched = true;
updateBoudingRect();
_Valid = splitDone;
//TTicks endTime = CTime::getPerformanceTime();
//nlinfo("%d ms for CCtrlPolygon::setVertices", (int) (1000 * CTime::ticksToSecond(endTime - startTime)));
}
static inline bool totallyInside(const CVector &minCorner, const CVector &maxCorner, sint32 cx, sint32 cy, sint32 cw, sint32 ch)
{
return (sint32) maxCorner.x < (cx + cw) &&
(sint32) minCorner.x >= cx &&
(sint32) maxCorner.y < (cy + ch) &&
(sint32) minCorner.y >= cy;
}
static inline bool totallyOutside(const CVector &minCorner, const CVector &maxCorner, sint32 cx, sint32 cy, sint32 cw, sint32 ch)
{
return (sint32) minCorner.x >= (cx + cw) ||
(sint32) maxCorner.x < cx ||
(sint32) minCorner.y >= (cy + ch) ||
(sint32) maxCorner.y < cy;
}
// *********************************************************************************
/*void CCtrlPolygon::setMatrix(const NLMISC::CMatrix &mat)
{
const float *lhs = mat.get();
const float *rhs = _Matrix.get();
if (std::equal(lhs, lhs + 16, rhs)) return; // unmodified...
_Matrix = mat;
updateBoudingRect();
_Touched = true;
}*/
// *********************************************************************************
void CCtrlPolygon::draw()
{
H_AUTO(Rz_CCtrlPolygon_draw)
if (_Tris.empty()) return;
if (!_Parent) return;
CViewRenderer &vr = *CViewRenderer::getInstance();
if (_Touched)
{
_RealTris.clear();
uint numTris = (uint)_Tris.size();
sint32 cornerX, cornerY;
static std::vector winTris;
winTris.resize(numTris);
_Parent->getCorner(cornerX, cornerY, _ParentPosRef);
/*CMatrix m = _Matrix;
m.setPos(m.getPos() + CVector((float) cornerX, (float) cornerY, 0.f));*/
for(uint k = 0; k < numTris; ++k)
{
/*winTris[k].V0 = m * _Tris[k].V0;
winTris[k].V1 = m * _Tris[k].V1;
winTris[k].V2 = m * _Tris[k].V2;*/
CVector2f result;
computeScaledVertex(result, _Tris[k].V0);
winTris[k].V0.set(result.x + cornerX, result.y + cornerY, 0.f);
computeScaledVertex(result, _Tris[k].V1);
winTris[k].V1.set(result.x + cornerX, result.y + cornerY, 0.f);
computeScaledVertex(result, _Tris[k].V2);
winTris[k].V2.set(result.x + cornerX, result.y + cornerY, 0.f);
}
// recompute & reclip poly
_RealTris.clear();
sint32 cx, cy, cw, ch;
vr.getClipWindow(cx, cy, cw, ch);
// per tri clip
NLMISC::CVector minCorner;
NLMISC::CVector maxCorner;
for(uint k = 0; k < numTris; ++k)
{
winTris[k].getMinCorner(minCorner);
winTris[k].getMaxCorner(maxCorner);
if (totallyOutside(minCorner, maxCorner, cx, cy, cw, ch)) continue;
if (totallyInside(minCorner, maxCorner, cx, cy, cw, ch))
{
_RealTris.push_back(winTris[k]);
}
else
{
const uint maxNumCorners = 8;
static CVector outPos0[maxNumCorners];
static CVector outPos1[maxNumCorners];
//
outPos0[0] = winTris[k].V0;
outPos0[1] = winTris[k].V1;
outPos0[2] = winTris[k].V2;
//
CVector *pPos0 = outPos0;
CVector *pPos1 = outPos1;
//
sint count = 3;
//
if ((sint32) minCorner.x < cx)
{
// clip left
CPlane clipper(-1.f, 0.f, 0.f, (float) cx);
count = clipper.clipPolygonBack(pPos0, pPos1, count);
std::swap(pPos0, pPos1);
}
if ((sint32) maxCorner.x > cx + cw)
{
// clip right
CPlane clipper(1.f, 0.f, 0.f, - (float) (cx + cw));
count = clipper.clipPolygonBack(pPos0, pPos1, count);
std::swap(pPos0, pPos1);
}
//
if ((sint32) minCorner.y < cy)
{
// clip bottom
CPlane clipper(0.f, -1.f, 0.f, (float) cy);
count = clipper.clipPolygonBack(pPos0, pPos1, count);
std::swap(pPos0, pPos1);
}
if ((sint32) maxCorner.y > cy + ch)
{
// clip top
CPlane clipper(0.f, 1.f, 0.f, - (float) (cy + ch));
count = clipper.clipPolygonBack(pPos0, pPos1, count);
std::swap(pPos0, pPos1);
}
nlassert(count <= 8);
if (count >= 3)
{
for(uint k = 0; k < (uint) (count - 2); ++k)
{
_RealTris.push_back(NLMISC::CTriangle(pPos0[0], pPos0[k + 1], pPos0[k + 2]));
}
}
}
}
_Touched = false;
}
if (_RealTris.empty()) return;
CRGBA col;
if(getModulateGlobalColor())
{
col.modulateFromColor (_Color, CWidgetManager::getInstance()->getGlobalColorForContent());
}
else
{
col= _Color;
col.A = (uint8)(((sint32)col.A*((sint32)CWidgetManager::getInstance()->getGlobalColorForContent().A+1))>>8);
}
vr.drawUnclippedTriangles(_RenderLayer, _RealTris, col);
}
// *********************************************************************************
void CCtrlPolygon::updateCoords()
{
H_AUTO(Rz_CCtrlPolygon_updateCoords)
CCtrlBase::updateCoords();
updateBoudingRect();
// assume that clipping will have to be done again, real update of triangle will be done at render time
_Touched = true;
}
// *********************************************************************************
void CCtrlPolygon::setAlpha(sint32 a)
{
H_AUTO(Rz_CCtrlPolygon_setAlpha)
_Color.A = (uint8) a;
}
// *********************************************************************************
bool CCtrlPolygon::handleEvent(const NLGUI::CEventDescriptor &/* event */)
{
H_AUTO(Rz_CCtrlPolygon_handleEvent)
return false;
}
// *********************************************************************************
// TMP TMP
void CCtrlPolygon::computeScaledVertex(NLMISC::CVector2f &dest, const NLMISC::CVector2f &src)
{
H_AUTO(Rz_CCtrlPolygon_computeScaledVertex)
dest.set(src.x, src.y);
}
// *********************************************************************************
// TMP TMP
void CCtrlPolygon::touch()
{
H_AUTO(Rz_CCtrlPolygon_touch)
updateBoudingRect();
_Touched = true;
}
}