1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 19:23:52 +00:00
openmw/apps/opencs/view/render/cellwater.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

195 lines
5.7 KiB
C++
Raw Normal View History

2016-08-05 02:58:55 +00:00
#include "cellwater.hpp"
2022-10-19 17:02:00 +00:00
#include <memory>
#include <string_view>
2016-08-05 02:58:55 +00:00
#include <osg/Geometry>
#include <osg/Group>
#include <osg/PositionAttitudeTransform>
2022-10-19 17:02:00 +00:00
#include <osg/StateAttribute>
#include <osg/StateSet>
#include <osg/Texture2D>
#include <osg/Texture>
#include <osg/Vec3d>
#include <osg/Vec3f>
#include <apps/opencs/model/world/collection.hpp>
#include <apps/opencs/model/world/idcollection.hpp>
#include <apps/opencs/model/world/record.hpp>
#include <apps/opencs/model/world/universalid.hpp>
2016-08-05 02:58:55 +00:00
#include <components/esm3/loadland.hpp>
2016-08-05 18:55:40 +00:00
#include <components/fallback/fallback.hpp>
#include <components/misc/strings/lower.hpp>
2016-08-05 18:55:40 +00:00
#include <components/resource/imagemanager.hpp>
#include <components/resource/resourcesystem.hpp>
#include <components/sceneutil/waterutil.hpp>
2016-08-05 02:58:55 +00:00
#include "../../model/world/cell.hpp"
#include "../../model/world/cellcoordinates.hpp"
#include "../../model/world/data.hpp"
#include "mask.hpp"
2016-08-05 02:58:55 +00:00
namespace CSVRender
{
const int CellWater::CellSize = ESM::Land::REAL_SIZE;
CellWater::CellWater(
CSMWorld::Data& data, osg::Group* cellNode, const std::string& id, const CSMWorld::CellCoordinates& cellCoords)
: mData(data)
, mId(id)
, mParentNode(cellNode)
2020-11-13 07:39:47 +00:00
, mWaterTransform(nullptr)
, mWaterGroup(nullptr)
2020-11-13 07:39:47 +00:00
, mWaterGeometry(nullptr)
, mDeleted(false)
2016-08-05 02:58:55 +00:00
, mExterior(false)
, mHasWater(false)
{
mWaterTransform = new osg::PositionAttitudeTransform();
mWaterTransform->setPosition(osg::Vec3f(
cellCoords.getX() * CellSize + CellSize / 2.f, cellCoords.getY() * CellSize + CellSize / 2.f, 0));
mWaterTransform->setNodeMask(Mask_Water);
2016-08-05 02:58:55 +00:00
mParentNode->addChild(mWaterTransform);
mWaterGroup = new osg::Group();
mWaterTransform->addChild(mWaterGroup);
2016-08-05 02:58:55 +00:00
const int cellIndex = mData.getCells().searchId(ESM::RefId::stringRefId(mId));
2016-08-05 02:58:55 +00:00
if (cellIndex > -1)
{
updateCellData(mData.getCells().getRecord(cellIndex));
2016-08-05 02:58:55 +00:00
}
2016-12-14 15:39:33 +00:00
// Keep water existence/height up to date
2016-08-05 02:58:55 +00:00
QAbstractItemModel* cells = mData.getTableModel(CSMWorld::UniversalId::Type_Cells);
connect(cells, &QAbstractItemModel::dataChanged, this, &CellWater::cellDataChanged);
2016-08-05 02:58:55 +00:00
}
CellWater::~CellWater()
{
mParentNode->removeChild(mWaterTransform);
}
void CellWater::updateCellData(const CSMWorld::Record<CSMWorld::Cell>& cellRecord)
2016-08-05 02:58:55 +00:00
{
mDeleted = cellRecord.isDeleted();
if (!mDeleted)
2016-08-05 02:58:55 +00:00
{
const CSMWorld::Cell& cell = cellRecord.get();
2016-08-05 02:58:55 +00:00
if (mExterior != cell.isExterior() || mHasWater != cell.hasWater())
2016-08-05 02:58:55 +00:00
{
mExterior = cellRecord.get().isExterior();
mHasWater = cellRecord.get().hasWater();
recreate();
}
float waterHeight = -1;
if (!mExterior)
{
waterHeight = cellRecord.get().mWater;
2016-08-05 02:58:55 +00:00
}
osg::Vec3d pos = mWaterTransform->getPosition();
pos.z() = waterHeight;
mWaterTransform->setPosition(pos);
2016-08-05 02:58:55 +00:00
}
else
{
recreate();
2016-08-05 02:58:55 +00:00
}
}
2017-08-21 00:55:56 +00:00
void CellWater::reloadAssets()
{
recreate();
}
2016-08-05 02:58:55 +00:00
void CellWater::cellDataChanged(const QModelIndex& topLeft, const QModelIndex& bottomRight)
{
const CSMWorld::Collection<CSMWorld::Cell>& cells = mData.getCells();
int rowStart = -1;
int rowEnd = -1;
if (topLeft.parent().isValid())
{
rowStart = topLeft.parent().row();
rowEnd = bottomRight.parent().row();
}
else
{
rowStart = topLeft.row();
rowEnd = bottomRight.row();
}
for (int row = rowStart; row <= rowEnd; ++row)
{
const CSMWorld::Record<CSMWorld::Cell>& cellRecord = cells.getRecord(row);
2016-08-05 02:58:55 +00:00
if (cellRecord.get().mId == ESM::RefId::stringRefId(mId))
updateCellData(cellRecord);
2016-08-05 02:58:55 +00:00
}
}
void CellWater::recreate()
2016-08-05 02:58:55 +00:00
{
const int InteriorScalar = 20;
const int SegmentsPerCell = 1;
const int TextureRepeatsPerCell = 6;
2016-08-05 02:58:55 +00:00
const float Alpha = 0.5f;
2016-08-05 02:58:55 +00:00
const int RenderBin = osg::StateSet::TRANSPARENT_BIN - 1;
2016-08-05 02:58:55 +00:00
if (mWaterGeometry)
{
mWaterGroup->removeChild(mWaterGeometry);
2020-11-13 07:39:47 +00:00
mWaterGeometry = nullptr;
2016-08-05 02:58:55 +00:00
}
if (mDeleted || !mHasWater)
return;
float size;
int segments;
float textureRepeats;
2016-08-05 02:58:55 +00:00
if (mExterior)
2016-08-05 02:58:55 +00:00
{
size = CellSize;
segments = SegmentsPerCell;
textureRepeats = TextureRepeatsPerCell;
2016-08-05 02:58:55 +00:00
}
else
2016-08-05 02:58:55 +00:00
{
size = CellSize * InteriorScalar;
segments = SegmentsPerCell * InteriorScalar;
textureRepeats = TextureRepeatsPerCell * InteriorScalar;
2016-08-05 02:58:55 +00:00
}
mWaterGeometry = SceneUtil::createWaterGeometry(size, segments, textureRepeats);
mWaterGeometry->setStateSet(SceneUtil::createSimpleWaterStateSet(Alpha, RenderBin));
2016-08-05 02:58:55 +00:00
2016-08-05 18:55:40 +00:00
// Add water texture
2022-08-28 15:20:49 +00:00
std::string textureName = "textures/water/";
textureName += Fallback::Map::getString("Water_SurfaceTexture");
textureName += "00.dds";
2016-08-05 18:55:40 +00:00
Resource::ImageManager* imageManager = mData.getResourceSystem()->getImageManager();
osg::ref_ptr<osg::Texture2D> waterTexture = new osg::Texture2D();
waterTexture->setImage(imageManager->getImage(textureName));
waterTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::REPEAT);
waterTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::REPEAT);
mWaterGeometry->getStateSet()->setTextureAttributeAndModes(0, waterTexture, osg::StateAttribute::ON);
mWaterGroup->addChild(mWaterGeometry);
2016-08-05 02:58:55 +00:00
}
}