Terrain texture selection, support for vertex selection
parent
2e05e0e829
commit
d6722c7492
@ -0,0 +1,271 @@
|
|||||||
|
#include "terrainselection.hpp"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
#include <osg/Group>
|
||||||
|
#include <osg/Geometry>
|
||||||
|
#include <osg/PositionAttitudeTransform>
|
||||||
|
|
||||||
|
#include <components/esm/loadland.hpp>
|
||||||
|
|
||||||
|
#include "../../model/world/cellcoordinates.hpp"
|
||||||
|
#include "../../model/world/columnimp.hpp"
|
||||||
|
#include "../../model/world/idtable.hpp"
|
||||||
|
|
||||||
|
#include "cell.hpp"
|
||||||
|
#include "worldspacewidget.hpp"
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
const int cellSize {ESM::Land::REAL_SIZE};
|
||||||
|
const int landSize {ESM::Land::LAND_SIZE};
|
||||||
|
const int landTextureSize {ESM::Land::LAND_TEXTURE_SIZE};
|
||||||
|
}
|
||||||
|
|
||||||
|
CSVRender::TerrainSelection::TerrainSelection(osg::Group* parentNode, WorldspaceWidget *worldspaceWidget, TerrainSelectionType type):
|
||||||
|
mParentNode(parentNode), mWorldspaceWidget (worldspaceWidget), mDraggedOperationFlag(false), mSelectionType(type)
|
||||||
|
{
|
||||||
|
mGeometry = new osg::Geometry();
|
||||||
|
|
||||||
|
mSelectionNode = new osg::Group();
|
||||||
|
mSelectionNode->addChild(mGeometry);
|
||||||
|
|
||||||
|
activate();
|
||||||
|
}
|
||||||
|
|
||||||
|
CSVRender::TerrainSelection::~TerrainSelection()
|
||||||
|
{
|
||||||
|
deactivate();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::pair<int, int>> CSVRender::TerrainSelection::getTerrainSelection() const
|
||||||
|
{
|
||||||
|
return mSelection;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::onlySelect(const std::vector<std::pair<int, int>> localPositions)
|
||||||
|
{
|
||||||
|
mSelection.clear();
|
||||||
|
for(auto const& value: localPositions)
|
||||||
|
{
|
||||||
|
mSelection.emplace_back(value);
|
||||||
|
}
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::addSelect(const std::pair<int, int> localPos)
|
||||||
|
{
|
||||||
|
if (std::find(mSelection.begin(), mSelection.end(), localPos) == mSelection.end())
|
||||||
|
mSelection.emplace_back(localPos);
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::toggleSelect(const std::vector<std::pair<int, int>> localPositions, bool toggleInProgress)
|
||||||
|
{
|
||||||
|
if (toggleInProgress == true)
|
||||||
|
{
|
||||||
|
for(auto const& localPos: localPositions)
|
||||||
|
{
|
||||||
|
auto iter = std::find(mSelection.begin(), mSelection.end(), localPos);
|
||||||
|
auto itertemp = std::find(mTemporarySelection.begin(), mTemporarySelection.end(), localPos);
|
||||||
|
mDraggedOperationFlag = true;
|
||||||
|
|
||||||
|
if (itertemp == mTemporarySelection.end())
|
||||||
|
{
|
||||||
|
if (iter != mSelection.end())
|
||||||
|
{
|
||||||
|
mSelection.erase(iter);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mSelection.emplace_back(localPos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mTemporarySelection.push_back(localPos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (mDraggedOperationFlag == false)
|
||||||
|
{
|
||||||
|
for(auto const& localPos: localPositions)
|
||||||
|
{
|
||||||
|
const auto iter = std::find(mSelection.begin(), mSelection.end(), localPos);
|
||||||
|
if (iter != mSelection.end())
|
||||||
|
{
|
||||||
|
mSelection.erase(iter);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mSelection.emplace_back(localPos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
mDraggedOperationFlag = false;
|
||||||
|
mTemporarySelection.clear();
|
||||||
|
}
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::activate()
|
||||||
|
{
|
||||||
|
mParentNode->addChild(mSelectionNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::deactivate()
|
||||||
|
{
|
||||||
|
mParentNode->removeChild(mSelectionNode);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::update()
|
||||||
|
{
|
||||||
|
mSelectionNode->removeChild(mGeometry);
|
||||||
|
mGeometry = new osg::Geometry();
|
||||||
|
|
||||||
|
const osg::ref_ptr<osg::Vec3Array> vertices (new osg::Vec3Array);
|
||||||
|
|
||||||
|
switch (mSelectionType)
|
||||||
|
{
|
||||||
|
case TerrainSelectionType::Texture : drawTextureSelection(vertices);
|
||||||
|
break;
|
||||||
|
case TerrainSelectionType::Shape : drawShapeSelection(vertices);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
mGeometry->setVertexArray(vertices);
|
||||||
|
osg::ref_ptr<osg::DrawArrays> drawArrays = new osg::DrawArrays(osg::PrimitiveSet::LINES);
|
||||||
|
drawArrays->setCount(vertices->size());
|
||||||
|
mGeometry->addPrimitiveSet(drawArrays);
|
||||||
|
mSelectionNode->addChild(mGeometry);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::drawShapeSelection(const osg::ref_ptr<osg::Vec3Array> vertices)
|
||||||
|
{
|
||||||
|
if (!mSelection.empty())
|
||||||
|
{
|
||||||
|
for (std::pair<int, int> localPos : mSelection)
|
||||||
|
{
|
||||||
|
int x (localPos.first);
|
||||||
|
int y (localPos.second);
|
||||||
|
|
||||||
|
float xWorldCoord(CSMWorld::CellCoordinates::vertexSelectionToWorldCoords(x));
|
||||||
|
float yWorldCoord(CSMWorld::CellCoordinates::vertexSelectionToWorldCoords(y));
|
||||||
|
|
||||||
|
osg::Vec3f pointXY(xWorldCoord, yWorldCoord, calculateLandHeight(x, y) + 2);
|
||||||
|
|
||||||
|
vertices->push_back(pointXY);
|
||||||
|
vertices->push_back(osg::Vec3f(xWorldCoord, CSMWorld::CellCoordinates::vertexSelectionToWorldCoords(y - 1), calculateLandHeight(x, y - 1) + 2));
|
||||||
|
vertices->push_back(pointXY);
|
||||||
|
vertices->push_back(osg::Vec3f(CSMWorld::CellCoordinates::vertexSelectionToWorldCoords(x - 1), yWorldCoord, calculateLandHeight(x - 1, y) + 2));
|
||||||
|
|
||||||
|
const auto north = std::find(mSelection.begin(), mSelection.end(), std::make_pair(x, y + 1));
|
||||||
|
if (north == mSelection.end())
|
||||||
|
{
|
||||||
|
vertices->push_back(pointXY);
|
||||||
|
vertices->push_back(osg::Vec3f(xWorldCoord, CSMWorld::CellCoordinates::vertexSelectionToWorldCoords(y + 1), calculateLandHeight(x, y + 1) + 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto east = std::find(mSelection.begin(), mSelection.end(), std::make_pair(x + 1, y));
|
||||||
|
if (east == mSelection.end())
|
||||||
|
{
|
||||||
|
vertices->push_back(pointXY);
|
||||||
|
vertices->push_back(osg::Vec3f(CSMWorld::CellCoordinates::vertexSelectionToWorldCoords(x + 1), yWorldCoord, calculateLandHeight(x + 1, y) + 2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CSVRender::TerrainSelection::drawTextureSelection(const osg::ref_ptr<osg::Vec3Array> vertices)
|
||||||
|
{
|
||||||
|
if (!mSelection.empty())
|
||||||
|
{
|
||||||
|
|
||||||
|
// Nudge selection by 1/4th of a texture size, similar how blendmaps are nudged
|
||||||
|
const float nudgePercentage = 0.25f;
|
||||||
|
const int nudgeOffset = (cellSize / landTextureSize) * nudgePercentage;
|
||||||
|
const int landHeightsNudge = (cellSize / landSize) / (landSize - 1); // Does this work with all land size configurations?
|
||||||
|
|
||||||
|
const int textureSizeToLandSizeModifier = (landSize - 1) / landTextureSize;
|
||||||
|
|
||||||
|
for (std::pair<int, int> localPos : mSelection)
|
||||||
|
{
|
||||||
|
int x (localPos.first);
|
||||||
|
int y (localPos.second);
|
||||||
|
|
||||||
|
// convert texture selection to global vertex coordinates at selection box corners
|
||||||
|
int x1 = x * textureSizeToLandSizeModifier + landHeightsNudge;
|
||||||
|
int x2 = x * textureSizeToLandSizeModifier + textureSizeToLandSizeModifier + landHeightsNudge;
|
||||||
|
int y1 = y * textureSizeToLandSizeModifier - landHeightsNudge;
|
||||||
|
int y2 = y * textureSizeToLandSizeModifier + textureSizeToLandSizeModifier - landHeightsNudge;
|
||||||
|
|
||||||
|
// Draw edges (check all sides, draw lines between vertices, +1 height to keep lines above ground)
|
||||||
|
// Check adjancent selections, draw lines only to edges of the selection
|
||||||
|
const auto north = std::find(mSelection.begin(), mSelection.end(), std::make_pair(x, y + 1));
|
||||||
|
if (north == mSelection.end())
|
||||||
|
{
|
||||||
|
for(int i = 1; i < (textureSizeToLandSizeModifier + 1); i++)
|
||||||
|
{
|
||||||
|
float drawPreviousX = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x)+(i-1)*(cellSize / (landSize - 1));
|
||||||
|
float drawCurrentX = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x)+i*(cellSize / (landSize - 1));
|
||||||
|
vertices->push_back(osg::Vec3f(drawPreviousX + nudgeOffset, CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y + 1) - nudgeOffset, calculateLandHeight(x1+(i-1), y2)+2));
|
||||||
|
vertices->push_back(osg::Vec3f(drawCurrentX + nudgeOffset, CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y + 1) - nudgeOffset, calculateLandHeight(x1+i, y2)+2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto south = std::find(mSelection.begin(), mSelection.end(), std::make_pair(x, y - 1));
|
||||||
|
if (south == mSelection.end())
|
||||||
|
{
|
||||||
|
for(int i = 1; i < (textureSizeToLandSizeModifier + 1); i++)
|
||||||
|
{
|
||||||
|
float drawPreviousX = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x)+(i-1)*(cellSize / (landSize - 1));
|
||||||
|
float drawCurrentX = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x)+i*(cellSize / (landSize - 1));
|
||||||
|
vertices->push_back(osg::Vec3f(drawPreviousX + nudgeOffset, CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y) - nudgeOffset, calculateLandHeight(x1+(i-1), y1)+2));
|
||||||
|
vertices->push_back(osg::Vec3f(drawCurrentX + nudgeOffset, CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y) - nudgeOffset, calculateLandHeight(x1+i, y1)+2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto east = std::find(mSelection.begin(), mSelection.end(), std::make_pair(x + 1, y));
|
||||||
|
if (east == mSelection.end())
|
||||||
|
{
|
||||||
|
for(int i = 1; i < (textureSizeToLandSizeModifier + 1); i++)
|
||||||
|
{
|
||||||
|
float drawPreviousY = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y)+(i-1)*(cellSize / (landSize - 1));
|
||||||
|
float drawCurrentY = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y)+i*(cellSize / (landSize - 1));
|
||||||
|
vertices->push_back(osg::Vec3f(CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x + 1) + nudgeOffset, drawPreviousY - nudgeOffset, calculateLandHeight(x2, y1+(i-1))+2));
|
||||||
|
vertices->push_back(osg::Vec3f(CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x + 1) + nudgeOffset, drawCurrentY - nudgeOffset, calculateLandHeight(x2, y1+i)+2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto west = std::find(mSelection.begin(), mSelection.end(), std::make_pair(x - 1, y));
|
||||||
|
if (west == mSelection.end())
|
||||||
|
{
|
||||||
|
for(int i = 1; i < (textureSizeToLandSizeModifier + 1); i++)
|
||||||
|
{
|
||||||
|
float drawPreviousY = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y)+(i-1)*(cellSize / (landSize - 1));
|
||||||
|
float drawCurrentY = CSMWorld::CellCoordinates::textureSelectionToWorldCoords(y)+i*(cellSize / (landSize - 1));
|
||||||
|
vertices->push_back(osg::Vec3f(CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x) + nudgeOffset, drawPreviousY - nudgeOffset, calculateLandHeight(x1, y1+(i-1))+2));
|
||||||
|
vertices->push_back(osg::Vec3f(CSMWorld::CellCoordinates::textureSelectionToWorldCoords(x) + nudgeOffset, drawCurrentY - nudgeOffset, calculateLandHeight(x1, y1+i)+2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int CSVRender::TerrainSelection::calculateLandHeight(int x, int y) // global vertex coordinates
|
||||||
|
{
|
||||||
|
int cellX = std::floor((1.0f*x / (landSize - 1)));
|
||||||
|
int cellY = std::floor((1.0f*y / (landSize - 1)));
|
||||||
|
int localX = x - cellX * (landSize - 1);
|
||||||
|
int localY = y - cellY * (landSize - 1);
|
||||||
|
|
||||||
|
std::string cellId = CSMWorld::CellCoordinates::generateId(cellX, cellY);
|
||||||
|
|
||||||
|
CSMDoc::Document& document = mWorldspaceWidget->getDocument();
|
||||||
|
CSMWorld::IdTable& landTable = dynamic_cast<CSMWorld::IdTable&> (
|
||||||
|
*document.getData().getTableModel (CSMWorld::UniversalId::Type_Land));
|
||||||
|
int landshapeColumn = landTable.findColumnIndex(CSMWorld::Columns::ColumnId_LandHeightsIndex);
|
||||||
|
const CSMWorld::LandHeightsColumn::DataType mPointer = landTable.data(landTable.getModelIndex(cellId, landshapeColumn)).value<CSMWorld::LandHeightsColumn::DataType>();
|
||||||
|
|
||||||
|
return mPointer[localY*landSize + localX];
|
||||||
|
}
|
@ -0,0 +1,75 @@
|
|||||||
|
#ifndef CSV_RENDER_TERRAINSELECTION_H
|
||||||
|
#define CSV_RENDER_TERRAINSELECTION_H
|
||||||
|
|
||||||
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <osg/Vec3d>
|
||||||
|
#include <osg/ref_ptr>
|
||||||
|
#include <osg/PositionAttitudeTransform>
|
||||||
|
|
||||||
|
#include <components/esm/loadland.hpp>
|
||||||
|
#include "../../model/world/cellcoordinates.hpp"
|
||||||
|
|
||||||
|
namespace osg
|
||||||
|
{
|
||||||
|
class Group;
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace CSVRender
|
||||||
|
{
|
||||||
|
struct WorldspaceHitResult;
|
||||||
|
class WorldspaceWidget;
|
||||||
|
|
||||||
|
enum class TerrainSelectionType
|
||||||
|
{
|
||||||
|
Texture,
|
||||||
|
Shape
|
||||||
|
};
|
||||||
|
|
||||||
|
/// \brief Class handling the terrain selection data and rendering
|
||||||
|
class TerrainSelection
|
||||||
|
{
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
TerrainSelection(osg::Group* parentNode, WorldspaceWidget *worldspaceWidget, TerrainSelectionType type);
|
||||||
|
~TerrainSelection();
|
||||||
|
|
||||||
|
void onlySelect(const std::vector<std::pair<int, int>> localPositions);
|
||||||
|
void addSelect(const std::pair<int, int> localPos);
|
||||||
|
void toggleSelect(const std::vector<std::pair<int, int>> localPositions, bool);
|
||||||
|
|
||||||
|
void activate();
|
||||||
|
void deactivate();
|
||||||
|
|
||||||
|
std::vector<std::pair<int, int>> getTerrainSelection() const;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
|
||||||
|
void addToSelection(osg::Vec3d worldPos);
|
||||||
|
void toggleSelection(osg::Vec3d worldPos);
|
||||||
|
void deselect();
|
||||||
|
|
||||||
|
void update();
|
||||||
|
|
||||||
|
void drawShapeSelection(const osg::ref_ptr<osg::Vec3Array> vertices);
|
||||||
|
void drawTextureSelection(const osg::ref_ptr<osg::Vec3Array> vertices);
|
||||||
|
|
||||||
|
int calculateLandHeight(int x, int y);
|
||||||
|
|
||||||
|
private:
|
||||||
|
|
||||||
|
osg::Group* mParentNode;
|
||||||
|
WorldspaceWidget *mWorldspaceWidget;
|
||||||
|
osg::ref_ptr<osg::PositionAttitudeTransform> mBaseNode;
|
||||||
|
osg::ref_ptr<osg::Geometry> mGeometry;
|
||||||
|
osg::ref_ptr<osg::Group> mSelectionNode;
|
||||||
|
std::vector<std::pair<int, int>> mSelection; // Global terrain selection coordinate in either vertex or texture units
|
||||||
|
std::vector<std::pair<int, int>> mTemporarySelection; // Used during toggle to compare the most recent drag operation
|
||||||
|
bool mDraggedOperationFlag; //true during drag operation, false when click-operation
|
||||||
|
TerrainSelectionType mSelectionType;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue