axmol/extensions/Particle3D/PU/CCPUBillboardChain.cpp

809 lines
29 KiB
C++

/****************************************************************************
Copyright (C) 2013 Henry van Merode. All rights reserved.
Copyright (c) 2015-2016 Chukong Technologies Inc.
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
https://axis-project.github.io/
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 "extensions/Particle3D/PU/CCPUBillboardChain.h"
#include <stddef.h> // offsetof
#include "base/ccTypes.h"
#include "extensions/Particle3D/PU/CCPUParticleSystem3D.h"
#include "base/CCDirector.h"
#include "renderer/ccShaders.h"
#include "renderer/CCMeshCommand.h"
#include "renderer/CCRenderer.h"
#include "renderer/CCTextureCache.h"
#include "renderer/backend/Types.h"
#include "renderer/backend/Buffer.h"
#include "renderer/backend/Device.h"
#include "2d/CCCamera.h"
#include "3d/CCMeshRenderer.h"
NS_AX_BEGIN
const size_t PUBillboardChain::SEGMENT_EMPTY = std::numeric_limits<size_t>::max();
//-----------------------------------------------------------------------
PUBillboardChain::Element::Element() {}
//-----------------------------------------------------------------------
PUBillboardChain::Element::Element(const Vec3& pos, float w, float tex, const Vec4& col, const Quaternion& ori)
: position(pos), width(w), texCoord(tex), color(col), orientation(ori)
{}
//-----------------------------------------------------------------------
PUBillboardChain::PUBillboardChain(std::string_view /*name*/,
std::string_view texFile,
size_t maxElements,
size_t numberOfChains,
bool useTextureCoords,
bool useColours,
bool dynamic)
: _maxElementsPerChain(maxElements)
, _chainCount(numberOfChains)
, _useTexCoords(useTextureCoords)
, _useVertexColour(useColours)
, _dynamic(dynamic)
, _vertexDeclDirty(true)
, _buffersNeedRecreating(true)
, _boundsDirty(true)
, _indexContentDirty(true)
, _vertexContentDirty(true)
, _texCoordDir(TCD_U)
, _faceCamera(true)
, _normalBase(Vec3::UNIT_X)
, _texture(nullptr)
, _programState(nullptr)
, _indexBuffer(nullptr)
, _vertexBuffer(nullptr)
, _texFile(texFile)
{
_stateBlock.setCullFace(false);
_stateBlock.setCullFaceSide(backend::CullMode::BACK);
_stateBlock.setDepthTest(false);
_stateBlock.setDepthWrite(false);
_stateBlock.setBlend(true);
_otherTexCoordRange[0] = 0.0f;
_otherTexCoordRange[1] = 1.0f;
setupChainContainers();
init(texFile);
}
//-----------------------------------------------------------------------
PUBillboardChain::~PUBillboardChain()
{
// CC_SAFE_RELEASE(_texture);
CC_SAFE_RELEASE(_programState);
CC_SAFE_RELEASE(_vertexBuffer);
CC_SAFE_RELEASE(_indexBuffer);
}
//-----------------------------------------------------------------------
void PUBillboardChain::setupChainContainers()
{
// Allocate enough space for everything
_chainElementList.resize(_chainCount * _maxElementsPerChain);
//_vertexData->vertexCount = _chainElementList.size() * 2;
// Configure chains
_chainSegmentList.resize(_chainCount);
for (size_t i = 0; i < _chainCount; ++i)
{
ChainSegment& seg = _chainSegmentList[i];
seg.start = i * _maxElementsPerChain;
seg.tail = seg.head = SEGMENT_EMPTY;
}
}
//-----------------------------------------------------------------------
void PUBillboardChain::setupVertexDeclaration()
{
// if (_vertexDeclDirty)
//{
// VertexDeclaration* decl = _vertexData->vertexDeclaration;
// decl->removeAllElements();
// size_t offset = 0;
// // Add a description for the buffer of the positions of the vertices
// decl->addElement(0, offset, VET_FLOAT3, VES_POSITION);
// offset += VertexElement::getTypeSize(VET_FLOAT3);
// if (_useVertexColour)
// {
// decl->addElement(0, offset, VET_COLOUR, VES_DIFFUSE);
// offset += VertexElement::getTypeSize(VET_COLOUR);
// }
// if (_useTexCoords)
// {
// decl->addElement(0, offset, VET_FLOAT2, VES_TEXTURE_COORDINATES);
// }
// if (!_useTexCoords && !_useVertexColour)
// {
// LogManager::getSingleton().logMessage(
// "Error - BillboardChain '" + mName + "' is using neither "
// "texture coordinates or vertex colours; it will not be "
// "visible on some rendering APIs so you should change this "
// "so you use one or the other.");
// }
// _vertexDeclDirty = false;
//}
}
//-----------------------------------------------------------------------
void PUBillboardChain::setupBuffers()
{
// setupVertexDeclaration();
if (_buffersNeedRecreating)
{
CC_SAFE_RELEASE_NULL(_vertexBuffer);
CC_SAFE_RELEASE_NULL(_indexBuffer);
size_t stride = sizeof(VertexInfo);
_vertexBuffer = backend::Device::getInstance()->newBuffer(
stride * _chainElementList.size() * 2, backend::BufferType::VERTEX, backend::BufferUsage::DYNAMIC);
VertexInfo vi = {Vec3(0.0f, 0.0f, 0.0f), Vec2(0.0f, 0.0f), Vec4::ONE};
_vertices.resize(_chainElementList.size() * 2, vi);
_indexBuffer =
backend::Device::getInstance()->newBuffer(_chainCount * _maxElementsPerChain * 6 * sizeof(uint16_t),
backend::BufferType::VERTEX, backend::BufferUsage::DYNAMIC);
_indices.resize(_chainCount * _maxElementsPerChain * 6, 0);
//// Create the vertex buffer (always dynamic due to the camera adjust)
// HardwareVertexBufferSharedPtr pBuffer =
// HardwareBufferManager::getSingleton().createVertexBuffer(
// _vertexData->vertexDeclaration->getVertexSize(0),
// _vertexData->vertexCount,
// HardwareBuffer::HBU_DYNAMIC_WRITE_ONLY_DISCARDABLE);
//// (re)Bind the buffer
//// Any existing buffer will lose its reference count and be destroyed
//_vertexData->vertexBufferBinding->setBinding(0, pBuffer);
//_indexData->indexBuffer =
// HardwareBufferManager::getSingleton().createIndexBuffer(
// HardwareIndexBuffer::IT_16BIT,
// _chainCount * _maxElementsPerChain * 6, // max we can use
// _dynamic? HardwareBuffer::HBU_DYNAMIC_WRITE_ONLY : HardwareBuffer::HBU_STATIC_WRITE_ONLY);
//// NB we don't set the indexCount on IndexData here since we will
//// probably use less than the maximum number of indices
_buffersNeedRecreating = false;
}
}
//-----------------------------------------------------------------------
void PUBillboardChain::setMaxChainElements(size_t maxElements)
{
_maxElementsPerChain = maxElements;
setupChainContainers();
_buffersNeedRecreating = _indexContentDirty = _vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::setNumberOfChains(size_t numChains)
{
_chainCount = numChains;
setupChainContainers();
_buffersNeedRecreating = _indexContentDirty = _vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::setUseTextureCoords(bool use)
{
_useTexCoords = use;
_vertexDeclDirty = _buffersNeedRecreating = true;
_indexContentDirty = _vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::setTextureCoordDirection(PUBillboardChain::TexCoordDirection dir)
{
_texCoordDir = dir;
_vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::setOtherTextureCoordRange(float start, float end)
{
_otherTexCoordRange[0] = start;
_otherTexCoordRange[1] = end;
_vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::setUseVertexColours(bool use)
{
_useVertexColour = use;
_vertexDeclDirty = _buffersNeedRecreating = true;
_indexContentDirty = _vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::setDynamic(bool dyn)
{
_dynamic = dyn;
_buffersNeedRecreating = _indexContentDirty = _vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::addChainElement(size_t chainIndex, const PUBillboardChain::Element& dtls)
{
CCASSERT(chainIndex < _chainCount, "chainIndex out of bounds");
ChainSegment& seg = _chainSegmentList[chainIndex];
if (seg.head == SEGMENT_EMPTY)
{
// Tail starts at end, head grows backwards
seg.tail = _maxElementsPerChain - 1;
seg.head = seg.tail;
}
else
{
if (seg.head == 0)
{
// Wrap backwards
seg.head = _maxElementsPerChain - 1;
}
else
{
// Just step backward
--seg.head;
}
// Run out of elements?
if (seg.head == seg.tail)
{
// Move tail backwards too, losing the end of the segment and re-using
// it in the head
if (seg.tail == 0)
seg.tail = _maxElementsPerChain - 1;
else
--seg.tail;
}
}
// Set the details
_chainElementList[seg.start + seg.head] = dtls;
_vertexContentDirty = true;
_indexContentDirty = true;
_boundsDirty = true;
//// tell parent node to update bounds
// if (mParentNode)
// mParentNode->needUpdate();
}
//-----------------------------------------------------------------------
void PUBillboardChain::removeChainElement(size_t chainIndex)
{
CCASSERT(chainIndex < _chainCount, "chainIndex out of bounds");
ChainSegment& seg = _chainSegmentList[chainIndex];
if (seg.head == SEGMENT_EMPTY)
return; // do nothing, nothing to remove
if (seg.tail == seg.head)
{
// last item
seg.head = seg.tail = SEGMENT_EMPTY;
}
else if (seg.tail == 0)
{
seg.tail = _maxElementsPerChain - 1;
}
else
{
--seg.tail;
}
// we removed an entry so indexes need updating
_vertexContentDirty = true;
_indexContentDirty = true;
_boundsDirty = true;
//// tell parent node to update bounds
// if (mParentNode)
// mParentNode->needUpdate();
}
//-----------------------------------------------------------------------
void PUBillboardChain::clearChain(size_t chainIndex)
{
CCASSERT(chainIndex < _chainCount, "chainIndex out of bounds");
ChainSegment& seg = _chainSegmentList[chainIndex];
// Just reset head & tail
seg.tail = seg.head = SEGMENT_EMPTY;
// we removed an entry so indexes need updating
_vertexContentDirty = true;
_indexContentDirty = true;
_boundsDirty = true;
//// tell parent node to update bounds
// if (mParentNode)
// mParentNode->needUpdate();
}
//-----------------------------------------------------------------------
void PUBillboardChain::clearAllChains()
{
for (size_t i = 0; i < _chainCount; ++i)
{
clearChain(i);
}
}
//-----------------------------------------------------------------------
void PUBillboardChain::setFaceCamera(bool faceCamera, const Vec3& normalVector)
{
_faceCamera = faceCamera;
_normalBase = normalVector;
_normalBase.normalize();
_vertexContentDirty = true;
}
//-----------------------------------------------------------------------
void PUBillboardChain::updateChainElement(size_t chainIndex, size_t elementIndex, const PUBillboardChain::Element& dtls)
{
CCASSERT(chainIndex < _chainCount, "chainIndex out of bounds");
ChainSegment& seg = _chainSegmentList[chainIndex];
CCASSERT(seg.head != SEGMENT_EMPTY, "Chain segment is empty");
size_t idx = seg.head + elementIndex;
// adjust for the edge and start
idx = (idx % _maxElementsPerChain) + seg.start;
_chainElementList[idx] = dtls;
_vertexContentDirty = true;
_boundsDirty = true;
//// tell parent node to update bounds
// if (mParentNode)
// mParentNode->needUpdate();
}
//-----------------------------------------------------------------------
const PUBillboardChain::Element& PUBillboardChain::getChainElement(size_t chainIndex, size_t elementIndex) const
{
CCASSERT(chainIndex < _chainCount, "chainIndex out of bounds");
const ChainSegment& seg = _chainSegmentList[chainIndex];
size_t idx = seg.head + elementIndex;
// adjust for the edge and start
idx = (idx % _maxElementsPerChain) + seg.start;
return _chainElementList[idx];
}
//-----------------------------------------------------------------------
size_t PUBillboardChain::getNumChainElements(size_t chainIndex) const
{
CCASSERT(chainIndex < _chainCount, "chainIndex out of bounds");
const ChainSegment& seg = _chainSegmentList[chainIndex];
if (seg.tail < seg.head)
{
return seg.tail - seg.head + _maxElementsPerChain + 1;
}
else
{
return seg.tail - seg.head + 1;
}
}
//-----------------------------------------------------------------------
void PUBillboardChain::updateVertexBuffer(const Mat4& camMat)
{
setupBuffers();
// The contents of the vertex buffer are correct if they are not dirty
// and the camera used to build the vertex buffer is still the current
// camera.
if (!_vertexContentDirty)
return;
VertexInfo vi = {Vec3(0.0f, 0.0f, 0.0f), Vec2(0.0f, 0.0f), Vec4::ONE};
_vertices.assign(_vertices.size(), vi);
// HardwareVertexBufferSharedPtr pBuffer =
// _vertexData->vertexBufferBinding->getBuffer(0);
// void* pBufferStart = pBuffer->lock(HardwareBuffer::HBL_DISCARD);
// const Vector3& camPos = cam->getDerivedPosition();
// Vector3 eyePos = mParentNode->_getDerivedOrientation().Inverse() *
// (camPos - mParentNode->_getDerivedPosition()) / mParentNode->_getDerivedScale();
Vec3 eyePos(camMat.m[12], camMat.m[13], camMat.m[14]);
Vec3 chainTangent;
for (ChainSegmentList::iterator segi = _chainSegmentList.begin(); segi != _chainSegmentList.end(); ++segi)
{
ChainSegment& seg = *segi;
// Skip 0 or 1 element segment counts
if (seg.head != SEGMENT_EMPTY && seg.head != seg.tail)
{
size_t laste = seg.head;
for (size_t e = seg.head;; ++e) // until break
{
// Wrap forwards
if (e == _maxElementsPerChain)
e = 0;
Element& elem = _chainElementList[e + seg.start];
CCASSERT(((e + seg.start) * 2) < 65536, "Too many elements!");
unsigned short vertexIndex = static_cast<unsigned short>((e + seg.start) * 2);
//// Determine base pointer to vertex #1
// void* pBase = static_cast<void*>(
// static_cast<char*>(pBufferStart) +
// pBuffer->getVertexSize() * baseIdx);
// Get index of next item
size_t nexte = e + 1;
if (nexte == _maxElementsPerChain)
nexte = 0;
if (e == seg.head)
{
// No laste, use next item
chainTangent = _chainElementList[nexte + seg.start].position - elem.position;
}
else if (e == seg.tail)
{
// No nexte, use only last item
chainTangent = elem.position - _chainElementList[laste + seg.start].position;
}
else
{
// A mid position, use tangent across both prev and next
chainTangent =
_chainElementList[nexte + seg.start].position - _chainElementList[laste + seg.start].position;
}
Vec3 vP1ToEye;
// if( _faceCamera )
vP1ToEye = eyePos - elem.position;
// else
// vP1ToEye = elem.orientation * _normalBase;
Vec3 vPerpendicular;
Vec3::cross(chainTangent, vP1ToEye, &vPerpendicular);
vPerpendicular.normalize();
vPerpendicular *= (elem.width * 0.5f);
Vec3 pos0 = elem.position - vPerpendicular;
Vec3 pos1 = elem.position + vPerpendicular;
// float* pFloat = static_cast<float*>(pBase);
//// pos1
//*pFloat++ = pos0.x;
//*pFloat++ = pos0.y;
//*pFloat++ = pos0.z;
_vertices[vertexIndex + 0].position = pos0;
// pBase = static_cast<void*>(pFloat);
if (_useVertexColour)
{
// RGBA* pCol = static_cast<RGBA*>(pBase);
// Root::getSingleton().convertColourValue(elem.colour, pCol);
// pCol++;
// pBase = static_cast<void*>(pCol);
_vertices[vertexIndex + 0].color = elem.color;
}
if (_useTexCoords)
{
// pFloat = static_cast<float*>(pBase);
if (_texCoordDir == TCD_U)
{
//*pFloat++ = elem.texCoord;
//*pFloat++ = _otherTexCoordRange[0];
_vertices[vertexIndex + 0].uv.x = elem.texCoord;
_vertices[vertexIndex + 0].uv.y = _otherTexCoordRange[0];
}
else
{
//*pFloat++ = _otherTexCoordRange[0];
//*pFloat++ = elem.texCoord;
_vertices[vertexIndex + 0].uv.x = _otherTexCoordRange[0];
_vertices[vertexIndex + 0].uv.y = elem.texCoord;
}
// pBase = static_cast<void*>(pFloat);
}
// pos2
// pFloat = static_cast<float*>(pBase);
//*pFloat++ = pos1.x;
//*pFloat++ = pos1.y;
//*pFloat++ = pos1.z;
// pBase = static_cast<void*>(pFloat);
_vertices[vertexIndex + 1].position = pos1;
if (_useVertexColour)
{
// RGBA* pCol = static_cast<RGBA*>(pBase);
// Root::getSingleton().convertColourValue(elem.colour, pCol);
// pCol++;
// pBase = static_cast<void*>(pCol);
_vertices[vertexIndex + 1].color = elem.color;
}
if (_useTexCoords)
{
// pFloat = static_cast<float*>(pBase);
if (_texCoordDir == TCD_U)
{
//*pFloat++ = elem.texCoord;
//*pFloat++ = _otherTexCoordRange[1];
_vertices[vertexIndex + 1].uv.x = elem.texCoord;
_vertices[vertexIndex + 1].uv.y = _otherTexCoordRange[1];
}
else
{
//*pFloat++ = _otherTexCoordRange[1];
//*pFloat++ = elem.texCoord;
_vertices[vertexIndex + 1].uv.x = _otherTexCoordRange[1];
_vertices[vertexIndex + 1].uv.y = elem.texCoord;
}
}
if (e == seg.tail)
break; // last one
laste = e;
// vertexIndex += 2;
} // element
} // segment valid?
} // each segment
_vertexBuffer->updateData(&_vertices[0], sizeof(_vertices[0]) * _vertices.size());
// pBuffer->unlock();
//_vertexCameraUsed = cam;
_vertexContentDirty = false;
}
//-----------------------------------------------------------------------
void PUBillboardChain::updateIndexBuffer()
{
setupBuffers();
if (_indexContentDirty)
{
// uint16* pShort = static_cast<uint16*>(
// _indexData->indexBuffer->lock(HardwareBuffer::HBL_DISCARD));
//_indexData->indexCount = 0;
// indexes
unsigned short index = 0;
for (ChainSegmentList::iterator segi = _chainSegmentList.begin(); segi != _chainSegmentList.end(); ++segi)
{
ChainSegment& seg = *segi;
// Skip 0 or 1 element segment counts
if (seg.head != SEGMENT_EMPTY && seg.head != seg.tail)
{
// Start from head + 1 since it's only useful in pairs
size_t laste = seg.head;
while (1) // until break
{
size_t e = laste + 1;
// Wrap forwards
if (e == _maxElementsPerChain)
e = 0;
// indexes of this element are (e * 2) and (e * 2) + 1
// indexes of the last element are the same, -2
CCASSERT(((e + seg.start) * 2) < 65536, "Too many elements!");
unsigned short baseIdx = static_cast<unsigned short>((e + seg.start) * 2);
unsigned short lastBaseIdx = static_cast<unsigned short>((laste + seg.start) * 2);
//*pShort++ = lastBaseIdx;
//*pShort++ = lastBaseIdx + 1;
//*pShort++ = baseIdx;
//*pShort++ = lastBaseIdx + 1;
//*pShort++ = baseIdx + 1;
//*pShort++ = baseIdx;
//_indexData->indexCount += 6;
_indices[index++] = lastBaseIdx;
_indices[index++] = lastBaseIdx + 1;
_indices[index++] = baseIdx;
_indices[index++] = lastBaseIdx + 1;
_indices[index++] = baseIdx + 1;
_indices[index++] = baseIdx;
if (e == seg.tail)
break; // last one
laste = e;
}
}
}
_indexBuffer->updateData(&_indices[0], sizeof(_indices[0]) * _indices.size());
//_indexData->indexBuffer->unlock();
_indexContentDirty = false;
}
}
//-----------------------------------------------------------------------
void PUBillboardChain::init(std::string_view texFile)
{
CC_SAFE_RELEASE_NULL(_programState);
if (!texFile.empty())
{
auto tex = Director::getInstance()->getTextureCache()->addImage(texFile);
if (tex)
{
_texture = tex;
auto* program = backend::Program::getBuiltinProgram(backend::ProgramType::PARTICLE_TEXTURE_3D);
_programState = new backend::ProgramState(program);
}
}
if (!_programState)
{
auto* program = backend::Program::getBuiltinProgram(backend::ProgramType::PARTICLE_COLOR_3D);
_programState = new backend::ProgramState(program);
}
auto& pipelineDescriptor = _meshCommand.getPipelineDescriptor();
pipelineDescriptor.programState = _programState;
auto layout = _programState->getVertexLayout();
const auto& attributeInfo = _programState->getProgram()->getActiveAttributes();
auto iter = attributeInfo.find("a_position");
if (iter != attributeInfo.end())
{
layout->setAttribute("a_position", iter->second.location, backend::VertexFormat::FLOAT3,
offsetof(VertexInfo, position), false);
}
iter = attributeInfo.find("a_texCoord");
if (iter != attributeInfo.end())
{
layout->setAttribute("a_texCoord", iter->second.location, backend::VertexFormat::FLOAT2,
offsetof(VertexInfo, uv), false);
}
iter = attributeInfo.find("a_color");
if (iter != attributeInfo.end())
{
layout->setAttribute("a_color", iter->second.location, backend::VertexFormat::FLOAT4,
offsetof(VertexInfo, color), false);
}
layout->setLayout(sizeof(VertexInfo));
_locColor = _programState->getUniformLocation("u_color");
_locPMatrix = _programState->getUniformLocation("u_PMatrix");
_locTexture = _programState->getUniformLocation("u_tex0");
_meshCommand.setTransparent(true);
_meshCommand.setSkipBatching(true);
_stateBlock.setDepthTest(true);
_stateBlock.setDepthWrite(false);
_stateBlock.setCullFaceSide(backend::CullMode::BACK);
_stateBlock.setCullFace(true);
_meshCommand.setBeforeCallback(CC_CALLBACK_0(PUBillboardChain::onBeforeDraw, this));
_meshCommand.setAfterCallback(CC_CALLBACK_0(PUBillboardChain::onAfterDraw, this));
}
void PUBillboardChain::render(Renderer* renderer, const Mat4& transform, ParticleSystem3D* particleSystem)
{
auto camera = Camera::getVisitingCamera();
auto cameraMat = camera->getNodeToWorldTransform();
if (!_chainSegmentList.empty())
{
updateVertexBuffer(cameraMat);
updateIndexBuffer();
_meshCommand.setVertexBuffer(_vertexBuffer);
_meshCommand.setIndexBuffer(_indexBuffer, MeshCommand::IndexFormat::U_SHORT);
_meshCommand.setIndexDrawInfo(0, _indices.size());
if (!_vertices.empty() && !_indices.empty())
{
_meshCommand.init(0.0);
_stateBlock.setBlendFunc(particleSystem->getBlendFunc());
auto& projectionMatrix = Director::getInstance()->getMatrix(MATRIX_STACK_TYPE::MATRIX_STACK_PROJECTION);
_programState->setUniform(_locPMatrix, &projectionMatrix.m, sizeof(projectionMatrix.m));
if (_texture)
{
_programState->setTexture(_locTexture, 0, _texture->getBackendTexture());
}
auto uColor = Vec4(1, 1, 1, 1);
_programState->setUniform(_locColor, &uColor, sizeof(uColor));
renderer->addCommand(&_meshCommand);
}
}
}
void PUBillboardChain::setDepthTest(bool isDepthTest)
{
_stateBlock.setDepthTest(isDepthTest);
}
void PUBillboardChain::setDepthWrite(bool isDepthWrite)
{
_stateBlock.setDepthWrite(isDepthWrite);
}
void PUBillboardChain::setBlendFunc(const BlendFunc& blendFunc)
{
_stateBlock.setBlendFunc(blendFunc);
}
void PUBillboardChain::onBeforeDraw()
{
auto* renderer = Director::getInstance()->getRenderer();
auto& pipelineDescriptor = _meshCommand.getPipelineDescriptor();
_rendererDepthTestEnabled = renderer->getDepthTest();
_rendererDepthCmpFunc = renderer->getDepthCompareFunction();
_rendererCullMode = renderer->getCullMode();
_rendererDepthWrite = renderer->getDepthWrite();
_rendererWinding = renderer->getWinding();
_stateBlock.bind(&pipelineDescriptor);
renderer->setDepthTest(true);
}
void PUBillboardChain::onAfterDraw()
{
auto* renderer = Director::getInstance()->getRenderer();
renderer->setDepthTest(_rendererDepthTestEnabled);
renderer->setDepthCompareFunction(_rendererDepthCmpFunc);
renderer->setCullMode(_rendererCullMode);
renderer->setDepthWrite(_rendererDepthWrite);
renderer->setWinding(_rendererWinding);
}
//-----------------------------------------------------------------------
// void PUBillboardChain::_updateRenderQueue(RenderQueue* queue)
//{
// updateIndexBuffer();
//
// if (mIndexData->indexCount > 0)
// {
// if (mRenderQueuePrioritySet)
// queue->addRenderable(this, mRenderQueueID, mRenderQueuePriority);
// else if (mRenderQueueIDSet)
// queue->addRenderable(this, mRenderQueueID);
// else
// queue->addRenderable(this);
// }
//
//}
//-----------------------------------------------------------------------
// void PUBillboardChain::getRenderOperation(RenderOperation& op)
//{
// op.indexData = mIndexData;
// op.operationType = RenderOperation::OT_TRIANGLE_LIST;
// op.srcRenderable = this;
// op.useIndexes = true;
// op.vertexData = mVertexData;
//}
//-----------------------------------------------------------------------
// bool PUBillboardChain::preRender(SceneManager* sm, RenderSystem* rsys)
//{
// // Retrieve the current viewport from the scene manager.
// // The viewport is only valid during a viewport update.
// Viewport *currentViewport = sm->getCurrentViewport();
// if( !currentViewport )
// return false;
//
// updateVertexBuffer(currentViewport->getCamera());
// return true;
//}
//-----------------------------------------------------------------------
// void PUBillboardChain::getWorldTransforms(Matrix4* xform) const
//{
// *xform = _getParentNodeFullTransform();
//}
//---------------------------------------------------------------------
NS_AX_END