2010-12-14 00:59:56 +01:00
|
|
|
/*
|
2013-02-24 18:40:43 +01:00
|
|
|
Minetest
|
2013-02-24 19:38:45 +01:00
|
|
|
Copyright (C) 2010-2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2010-12-14 00:59:56 +01:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
2012-06-05 16:56:56 +02:00
|
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
2010-12-14 00:59:56 +01:00
|
|
|
(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
|
2012-06-05 16:56:56 +02:00
|
|
|
GNU Lesser General Public License for more details.
|
2010-12-14 00:59:56 +01:00
|
|
|
|
2012-06-05 16:56:56 +02:00
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
2010-12-14 00:59:56 +01:00
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
2017-08-17 22:19:39 +02:00
|
|
|
#pragma once
|
2010-12-14 00:59:56 +01:00
|
|
|
|
2012-06-17 03:00:31 +02:00
|
|
|
#include "irrlichttypes.h"
|
|
|
|
#include "irr_v3d.h"
|
|
|
|
#include <ITexture.h>
|
2011-01-25 23:41:06 +01:00
|
|
|
#include <string>
|
2014-12-18 13:25:36 +01:00
|
|
|
#include <vector>
|
2017-07-02 20:29:58 +02:00
|
|
|
#include <SMaterial.h>
|
2015-03-09 14:32:11 +01:00
|
|
|
#include "util/numeric.h"
|
2019-07-25 00:01:25 +02:00
|
|
|
#include "config.h"
|
2010-12-14 00:59:56 +01:00
|
|
|
|
2019-07-25 00:01:25 +02:00
|
|
|
#if ENABLE_GLES
|
2017-12-06 21:36:29 +01:00
|
|
|
#include <IVideoDriver.h>
|
|
|
|
#endif
|
|
|
|
|
2011-11-14 20:41:30 +01:00
|
|
|
class IGameDef;
|
2015-07-16 15:36:48 +02:00
|
|
|
struct TileSpec;
|
|
|
|
struct TileDef;
|
2011-11-14 20:41:30 +01:00
|
|
|
|
2017-03-10 18:25:58 +01:00
|
|
|
typedef std::vector<video::SColor> Palette;
|
|
|
|
|
2011-05-21 10:07:03 +02:00
|
|
|
/*
|
|
|
|
tile.{h,cpp}: Texture handling stuff.
|
|
|
|
*/
|
|
|
|
|
2013-02-15 20:13:53 +01:00
|
|
|
/*
|
|
|
|
Find out the full path of an image by trying different filename
|
|
|
|
extensions.
|
|
|
|
|
|
|
|
If failed, return "".
|
|
|
|
|
|
|
|
TODO: Should probably be moved out from here, because things needing
|
|
|
|
this function do not need anything else from this header
|
|
|
|
*/
|
|
|
|
std::string getImagePath(std::string path);
|
|
|
|
|
2011-05-21 10:07:03 +02:00
|
|
|
/*
|
|
|
|
Gets the path to a texture by first checking if the texture exists
|
|
|
|
in texture_path and if not, using the data path.
|
|
|
|
|
|
|
|
Checks all supported extensions by replacing the original extension.
|
|
|
|
|
|
|
|
If not found, returns "".
|
|
|
|
|
|
|
|
Utilizes a thread-safe cache.
|
|
|
|
*/
|
2019-09-29 19:57:29 +02:00
|
|
|
std::string getTexturePath(const std::string &filename, bool *is_base_pack = nullptr);
|
2011-05-21 10:07:03 +02:00
|
|
|
|
2013-08-02 15:18:48 +02:00
|
|
|
void clearTextureNameCache();
|
|
|
|
|
2011-02-11 18:55:42 +01:00
|
|
|
/*
|
2011-11-14 20:41:30 +01:00
|
|
|
TextureSource creates and caches textures.
|
2011-02-11 18:55:42 +01:00
|
|
|
*/
|
2011-11-14 20:41:30 +01:00
|
|
|
|
2013-08-20 22:38:14 +02:00
|
|
|
class ISimpleTextureSource
|
|
|
|
{
|
|
|
|
public:
|
2017-08-15 20:30:30 +02:00
|
|
|
ISimpleTextureSource() = default;
|
|
|
|
|
|
|
|
virtual ~ISimpleTextureSource() = default;
|
|
|
|
|
2013-08-20 22:38:14 +02:00
|
|
|
virtual video::ITexture* getTexture(
|
2017-06-21 08:04:45 +02:00
|
|
|
const std::string &name, u32 *id = nullptr) = 0;
|
2013-08-20 22:38:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class ITextureSource : public ISimpleTextureSource
|
2011-02-11 18:55:42 +01:00
|
|
|
{
|
|
|
|
public:
|
2017-08-15 20:30:30 +02:00
|
|
|
ITextureSource() = default;
|
|
|
|
|
|
|
|
virtual ~ITextureSource() = default;
|
|
|
|
|
2013-08-20 22:38:14 +02:00
|
|
|
virtual u32 getTextureId(const std::string &name)=0;
|
|
|
|
virtual std::string getTextureName(u32 id)=0;
|
|
|
|
virtual video::ITexture* getTexture(u32 id)=0;
|
2013-06-24 04:17:50 +02:00
|
|
|
virtual video::ITexture* getTexture(
|
2017-06-21 08:04:45 +02:00
|
|
|
const std::string &name, u32 *id = nullptr)=0;
|
2015-03-31 02:04:19 +02:00
|
|
|
virtual video::ITexture* getTextureForMesh(
|
2017-06-21 08:04:45 +02:00
|
|
|
const std::string &name, u32 *id = nullptr) = 0;
|
2017-03-10 18:25:58 +01:00
|
|
|
/*!
|
|
|
|
* Returns a palette from the given texture name.
|
|
|
|
* The pointer is valid until the texture source is
|
|
|
|
* destructed.
|
|
|
|
* Should be called from the main thread.
|
|
|
|
*/
|
|
|
|
virtual Palette* getPalette(const std::string &name) = 0;
|
2012-11-30 21:19:50 +01:00
|
|
|
virtual bool isKnownSourceImage(const std::string &name)=0;
|
2014-07-15 09:07:52 +02:00
|
|
|
virtual video::ITexture* getNormalTexture(const std::string &name)=0;
|
2015-06-22 04:34:56 +02:00
|
|
|
virtual video::SColor getTextureAverageColor(const std::string &name)=0;
|
2015-08-18 23:59:44 +02:00
|
|
|
virtual video::ITexture *getShaderFlagsTexture(bool normalmap_present)=0;
|
2011-02-11 18:55:42 +01:00
|
|
|
};
|
|
|
|
|
2011-11-14 20:41:30 +01:00
|
|
|
class IWritableTextureSource : public ITextureSource
|
2011-02-10 01:13:03 +01:00
|
|
|
{
|
|
|
|
public:
|
2017-08-15 20:30:30 +02:00
|
|
|
IWritableTextureSource() = default;
|
|
|
|
|
|
|
|
virtual ~IWritableTextureSource() = default;
|
|
|
|
|
2013-08-20 22:38:14 +02:00
|
|
|
virtual u32 getTextureId(const std::string &name)=0;
|
|
|
|
virtual std::string getTextureName(u32 id)=0;
|
|
|
|
virtual video::ITexture* getTexture(u32 id)=0;
|
2013-06-24 04:17:50 +02:00
|
|
|
virtual video::ITexture* getTexture(
|
2017-06-21 08:04:45 +02:00
|
|
|
const std::string &name, u32 *id = nullptr)=0;
|
2012-11-30 21:19:50 +01:00
|
|
|
virtual bool isKnownSourceImage(const std::string &name)=0;
|
2011-02-10 01:13:03 +01:00
|
|
|
|
2011-11-14 20:41:30 +01:00
|
|
|
virtual void processQueue()=0;
|
2011-11-15 12:13:18 +01:00
|
|
|
virtual void insertSourceImage(const std::string &name, video::IImage *img)=0;
|
|
|
|
virtual void rebuildImagesAndTextures()=0;
|
2014-07-15 09:07:52 +02:00
|
|
|
virtual video::ITexture* getNormalTexture(const std::string &name)=0;
|
2015-06-22 04:34:56 +02:00
|
|
|
virtual video::SColor getTextureAverageColor(const std::string &name)=0;
|
2015-08-18 23:59:44 +02:00
|
|
|
virtual video::ITexture *getShaderFlagsTexture(bool normalmap_present)=0;
|
2011-02-10 01:13:03 +01:00
|
|
|
};
|
|
|
|
|
2017-06-26 20:11:17 +02:00
|
|
|
IWritableTextureSource *createTextureSource();
|
2011-11-14 20:41:30 +01:00
|
|
|
|
2019-07-25 00:01:25 +02:00
|
|
|
#if ENABLE_GLES
|
2019-07-25 00:11:30 +02:00
|
|
|
bool hasNPotSupport();
|
2017-12-06 21:36:29 +01:00
|
|
|
video::IImage * Align2Npot2(video::IImage * image, irr::video::IVideoDriver* driver);
|
2014-04-21 14:10:59 +02:00
|
|
|
#endif
|
|
|
|
|
2011-02-08 00:12:55 +01:00
|
|
|
enum MaterialType{
|
2012-12-02 01:57:14 +01:00
|
|
|
TILE_MATERIAL_BASIC,
|
2013-04-24 00:39:10 +02:00
|
|
|
TILE_MATERIAL_ALPHA,
|
2012-12-02 01:57:14 +01:00
|
|
|
TILE_MATERIAL_LIQUID_TRANSPARENT,
|
|
|
|
TILE_MATERIAL_LIQUID_OPAQUE,
|
2014-05-14 23:19:31 +02:00
|
|
|
TILE_MATERIAL_WAVING_LEAVES,
|
2017-06-07 19:52:38 +02:00
|
|
|
TILE_MATERIAL_WAVING_PLANTS,
|
2019-03-27 01:18:43 +01:00
|
|
|
TILE_MATERIAL_OPAQUE,
|
|
|
|
TILE_MATERIAL_WAVING_LIQUID_BASIC,
|
|
|
|
TILE_MATERIAL_WAVING_LIQUID_TRANSPARENT,
|
|
|
|
TILE_MATERIAL_WAVING_LIQUID_OPAQUE,
|
2020-06-16 21:48:31 +02:00
|
|
|
TILE_MATERIAL_PLAIN,
|
|
|
|
TILE_MATERIAL_PLAIN_ALPHA
|
2011-02-08 00:12:55 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Material flags
|
2012-03-13 18:56:12 +01:00
|
|
|
// Should backface culling be enabled?
|
2011-02-08 00:12:55 +01:00
|
|
|
#define MATERIAL_FLAG_BACKFACE_CULLING 0x01
|
2012-03-13 18:56:12 +01:00
|
|
|
// Should a crack be drawn?
|
|
|
|
#define MATERIAL_FLAG_CRACK 0x02
|
|
|
|
// Should the crack be drawn on transparent pixels (unset) or not (set)?
|
|
|
|
// Ignored if MATERIAL_FLAG_CRACK is not set.
|
|
|
|
#define MATERIAL_FLAG_CRACK_OVERLAY 0x04
|
2016-12-23 13:48:32 +01:00
|
|
|
#define MATERIAL_FLAG_ANIMATION 0x08
|
2017-08-15 20:30:30 +02:00
|
|
|
//#define MATERIAL_FLAG_HIGHLIGHTED 0x10
|
2015-08-05 22:52:32 +02:00
|
|
|
#define MATERIAL_FLAG_TILEABLE_HORIZONTAL 0x20
|
|
|
|
#define MATERIAL_FLAG_TILEABLE_VERTICAL 0x40
|
2011-02-08 00:12:55 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
This fully defines the looks of a tile.
|
|
|
|
The SMaterial of a tile is constructed according to this.
|
|
|
|
*/
|
2014-07-15 09:07:52 +02:00
|
|
|
struct FrameSpec
|
|
|
|
{
|
2017-08-15 20:30:30 +02:00
|
|
|
FrameSpec() = default;
|
|
|
|
|
2017-06-21 08:04:45 +02:00
|
|
|
u32 texture_id = 0;
|
|
|
|
video::ITexture *texture = nullptr;
|
|
|
|
video::ITexture *normal_texture = nullptr;
|
|
|
|
video::ITexture *flags_texture = nullptr;
|
2014-07-15 09:07:52 +02:00
|
|
|
};
|
|
|
|
|
2017-04-21 15:34:59 +02:00
|
|
|
#define MAX_TILE_LAYERS 2
|
|
|
|
|
|
|
|
//! Defines a layer of a tile.
|
|
|
|
struct TileLayer
|
2011-01-28 00:38:16 +01:00
|
|
|
{
|
2017-08-15 20:30:30 +02:00
|
|
|
TileLayer() = default;
|
2011-01-28 00:38:16 +01:00
|
|
|
|
2017-01-12 15:46:30 +01:00
|
|
|
/*!
|
2017-04-21 15:34:59 +02:00
|
|
|
* Two layers are equal if they can be merged.
|
2017-01-12 15:46:30 +01:00
|
|
|
*/
|
2017-04-21 15:34:59 +02:00
|
|
|
bool operator==(const TileLayer &other) const
|
2011-01-28 00:38:16 +01:00
|
|
|
{
|
2017-04-21 15:34:59 +02:00
|
|
|
return
|
2013-06-24 04:17:50 +02:00
|
|
|
texture_id == other.texture_id &&
|
2011-02-08 00:12:55 +01:00
|
|
|
material_type == other.material_type &&
|
2013-03-23 19:17:00 +01:00
|
|
|
material_flags == other.material_flags &&
|
2017-10-15 09:34:14 +02:00
|
|
|
color == other.color &&
|
|
|
|
scale == other.scale;
|
2011-01-28 00:38:16 +01:00
|
|
|
}
|
2012-03-13 18:56:12 +01:00
|
|
|
|
2017-01-12 15:46:30 +01:00
|
|
|
/*!
|
2017-04-21 15:34:59 +02:00
|
|
|
* Two tiles are not equal if they must have different vertices.
|
2017-01-12 15:46:30 +01:00
|
|
|
*/
|
2017-04-21 15:34:59 +02:00
|
|
|
bool operator!=(const TileLayer &other) const
|
2012-03-13 18:56:12 +01:00
|
|
|
{
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
2017-04-21 15:34:59 +02:00
|
|
|
|
2011-02-08 00:12:55 +01:00
|
|
|
// Sets everything else except the texture in the material
|
2011-09-19 06:37:24 +02:00
|
|
|
void applyMaterialOptions(video::SMaterial &material) const
|
2010-12-18 12:10:37 +01:00
|
|
|
{
|
2014-10-03 09:49:06 +02:00
|
|
|
switch (material_type) {
|
2017-06-07 19:52:38 +02:00
|
|
|
case TILE_MATERIAL_OPAQUE:
|
|
|
|
case TILE_MATERIAL_LIQUID_OPAQUE:
|
2019-03-27 01:18:43 +01:00
|
|
|
case TILE_MATERIAL_WAVING_LIQUID_OPAQUE:
|
2017-06-07 19:52:38 +02:00
|
|
|
material.MaterialType = video::EMT_SOLID;
|
|
|
|
break;
|
2012-12-02 01:57:14 +01:00
|
|
|
case TILE_MATERIAL_BASIC:
|
2017-04-21 15:34:59 +02:00
|
|
|
case TILE_MATERIAL_WAVING_LEAVES:
|
|
|
|
case TILE_MATERIAL_WAVING_PLANTS:
|
2019-03-27 01:18:43 +01:00
|
|
|
case TILE_MATERIAL_WAVING_LIQUID_BASIC:
|
2018-01-26 20:08:54 +01:00
|
|
|
material.MaterialTypeParam = 0.5;
|
2011-02-08 00:12:55 +01:00
|
|
|
material.MaterialType = video::EMT_TRANSPARENT_ALPHA_CHANNEL_REF;
|
2012-12-02 01:57:14 +01:00
|
|
|
break;
|
2013-04-24 00:39:10 +02:00
|
|
|
case TILE_MATERIAL_ALPHA:
|
2012-12-02 01:57:14 +01:00
|
|
|
case TILE_MATERIAL_LIQUID_TRANSPARENT:
|
2019-03-27 01:18:43 +01:00
|
|
|
case TILE_MATERIAL_WAVING_LIQUID_TRANSPARENT:
|
2017-01-12 15:46:30 +01:00
|
|
|
material.MaterialType = video::EMT_TRANSPARENT_ALPHA_CHANNEL;
|
2012-12-02 01:57:14 +01:00
|
|
|
break;
|
2017-08-19 22:23:47 +02:00
|
|
|
default:
|
|
|
|
break;
|
2012-12-02 01:57:14 +01:00
|
|
|
}
|
2017-08-15 20:30:30 +02:00
|
|
|
material.BackfaceCulling = (material_flags & MATERIAL_FLAG_BACKFACE_CULLING) != 0;
|
2015-08-20 03:39:26 +02:00
|
|
|
if (!(material_flags & MATERIAL_FLAG_TILEABLE_HORIZONTAL)) {
|
|
|
|
material.TextureLayer[0].TextureWrapU = video::ETC_CLAMP_TO_EDGE;
|
|
|
|
}
|
|
|
|
if (!(material_flags & MATERIAL_FLAG_TILEABLE_VERTICAL)) {
|
|
|
|
material.TextureLayer[0].TextureWrapV = video::ETC_CLAMP_TO_EDGE;
|
|
|
|
}
|
2012-12-02 01:57:14 +01:00
|
|
|
}
|
2014-05-14 23:19:31 +02:00
|
|
|
|
|
|
|
void applyMaterialOptionsWithShaders(video::SMaterial &material) const
|
2012-12-02 01:57:14 +01:00
|
|
|
{
|
2017-08-15 20:30:30 +02:00
|
|
|
material.BackfaceCulling = (material_flags & MATERIAL_FLAG_BACKFACE_CULLING) != 0;
|
2015-08-05 22:52:32 +02:00
|
|
|
if (!(material_flags & MATERIAL_FLAG_TILEABLE_HORIZONTAL)) {
|
|
|
|
material.TextureLayer[0].TextureWrapU = video::ETC_CLAMP_TO_EDGE;
|
|
|
|
material.TextureLayer[1].TextureWrapU = video::ETC_CLAMP_TO_EDGE;
|
|
|
|
}
|
|
|
|
if (!(material_flags & MATERIAL_FLAG_TILEABLE_VERTICAL)) {
|
|
|
|
material.TextureLayer[0].TextureWrapV = video::ETC_CLAMP_TO_EDGE;
|
|
|
|
material.TextureLayer[1].TextureWrapV = video::ETC_CLAMP_TO_EDGE;
|
|
|
|
}
|
2011-01-25 23:41:06 +01:00
|
|
|
}
|
2017-04-18 07:48:17 +02:00
|
|
|
|
2017-04-21 15:34:59 +02:00
|
|
|
bool isTileable() const
|
|
|
|
{
|
|
|
|
return (material_flags & MATERIAL_FLAG_TILEABLE_HORIZONTAL)
|
|
|
|
&& (material_flags & MATERIAL_FLAG_TILEABLE_VERTICAL);
|
|
|
|
}
|
|
|
|
|
2017-04-29 09:16:32 +02:00
|
|
|
// Ordered for size, please do not reorder
|
|
|
|
|
2017-06-21 08:04:45 +02:00
|
|
|
video::ITexture *texture = nullptr;
|
|
|
|
video::ITexture *normal_texture = nullptr;
|
|
|
|
video::ITexture *flags_texture = nullptr;
|
2017-04-21 15:34:59 +02:00
|
|
|
|
2017-06-21 08:04:45 +02:00
|
|
|
u32 shader_id = 0;
|
2017-04-29 09:16:32 +02:00
|
|
|
|
2017-06-21 08:04:45 +02:00
|
|
|
u32 texture_id = 0;
|
2017-04-29 09:16:32 +02:00
|
|
|
|
2017-06-21 08:04:45 +02:00
|
|
|
u16 animation_frame_length_ms = 0;
|
2020-07-10 12:12:19 +02:00
|
|
|
u16 animation_frame_count = 1;
|
2017-04-29 09:16:32 +02:00
|
|
|
|
2017-06-21 08:04:45 +02:00
|
|
|
u8 material_type = TILE_MATERIAL_BASIC;
|
|
|
|
u8 material_flags =
|
|
|
|
//0 // <- DEBUG, Use the one below
|
|
|
|
MATERIAL_FLAG_BACKFACE_CULLING |
|
|
|
|
MATERIAL_FLAG_TILEABLE_HORIZONTAL|
|
|
|
|
MATERIAL_FLAG_TILEABLE_VERTICAL;
|
2017-04-29 09:16:32 +02:00
|
|
|
|
2017-01-12 15:46:30 +01:00
|
|
|
//! If true, the tile has its own color.
|
2017-06-21 08:04:45 +02:00
|
|
|
bool has_color = false;
|
2017-04-18 07:48:17 +02:00
|
|
|
|
2020-07-10 12:11:44 +02:00
|
|
|
std::vector<FrameSpec> *frames = nullptr;
|
2017-04-29 09:16:32 +02:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* The color of the tile, or if the tile does not own
|
|
|
|
* a color then the color of the node owning this tile.
|
|
|
|
*/
|
|
|
|
video::SColor color;
|
2017-10-15 09:34:14 +02:00
|
|
|
|
|
|
|
u8 scale;
|
2010-12-18 12:10:37 +01:00
|
|
|
};
|
2017-04-21 15:34:59 +02:00
|
|
|
|
|
|
|
/*!
|
|
|
|
* Defines a face of a node. May have up to two layers.
|
|
|
|
*/
|
|
|
|
struct TileSpec
|
|
|
|
{
|
2018-08-08 15:53:06 +02:00
|
|
|
TileSpec() = default;
|
2017-06-26 20:11:17 +02:00
|
|
|
|
2017-04-21 15:34:59 +02:00
|
|
|
/*!
|
|
|
|
* Returns true if this tile can be merged with the other tile.
|
|
|
|
*/
|
|
|
|
bool isTileable(const TileSpec &other) const {
|
|
|
|
for (int layer = 0; layer < MAX_TILE_LAYERS; layer++) {
|
|
|
|
if (layers[layer] != other.layers[layer])
|
|
|
|
return false;
|
|
|
|
if (!layers[layer].isTileable())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return rotation == 0
|
|
|
|
&& rotation == other.rotation
|
|
|
|
&& emissive_light == other.emissive_light;
|
|
|
|
}
|
|
|
|
|
2017-10-15 09:34:14 +02:00
|
|
|
//! If true, the tile rotation is ignored.
|
|
|
|
bool world_aligned = false;
|
|
|
|
//! Tile rotation.
|
2017-06-21 08:04:45 +02:00
|
|
|
u8 rotation = 0;
|
2017-04-21 15:34:59 +02:00
|
|
|
//! This much light does the tile emit.
|
2017-06-21 08:04:45 +02:00
|
|
|
u8 emissive_light = 0;
|
2017-04-21 15:34:59 +02:00
|
|
|
//! The first is base texture, the second is overlay.
|
|
|
|
TileLayer layers[MAX_TILE_LAYERS];
|
|
|
|
};
|
2017-09-13 22:03:18 +02:00
|
|
|
|
2018-03-10 19:33:37 +01:00
|
|
|
std::vector<std::string> getTextureDirs();
|