aboutsummaryrefslogtreecommitdiff
path: root/src/mapblock_mesh.h
blob: 82268efd2948ad8e836dca639e34e06b343a6e74 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
/*
Minetest-c55
Copyright (C) 2010-2011 celeron55, Perttu Ahola <celeron55@gmail.com>

This program is free software; you can redistribute it and/or modify
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
(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 Lesser General Public License for more details.

You should have received a copy of the GNU Lesser General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/

#ifndef MAPBLOCK_MESH_HEADER
#define MAPBLOCK_MESH_HEADER

#include "irrlichttypes_extrabloated.h"
#include "tile.h"
#include "voxel.h"
#include <map>

class IGameDef;

/*
	Mesh making stuff
*/


class MapBlock;

struct MeshMakeData
{
	VoxelManipulator m_vmanip;
	v3s16 m_blockpos;
	v3s16 m_crack_pos_relative;
	bool m_smooth_lighting;
	IGameDef *m_gamedef;

	MeshMakeData(IGameDef *gamedef);

	/*
		Copy central data directly from block, and other data from
		parent of block.
	*/
	void fill(MapBlock *block);

	/*
		Set up with only a single node at (1,1,1)
	*/
	void fillSingleNode(MapNode *node);

	/*
		Set the (node) position of a crack
	*/
	void setCrack(int crack_level, v3s16 crack_pos);

	/*
		Enable or disable smooth lighting
	*/
	void setSmoothLighting(bool smooth_lighting);
};

/*
	Holds a mesh for a mapblock.

	Besides the SMesh*, this contains information used for animating
	the vertex positions, colors and texture coordinates of the mesh.
	For example:
	- cracks [implemented]
	- day/night transitions [implemented]
	- animated flowing liquids [not implemented]
	- animating vertex positions for e.g. axles [not implemented]
*/
class MapBlockMesh
{
public:
	// Builds the mesh given
	MapBlockMesh(MeshMakeData *data);
	~MapBlockMesh();

	// Main animation function, parameters:
	//   faraway: whether the block is far away from the camera (~50 nodes)
	//   time: the global animation time, 0 .. 60 (repeats every minute)
	//   daynight_ratio: 0 .. 1000
	//   crack: -1 .. CRACK_ANIMATION_LENGTH-1 (-1 for off)
	// Returns true if anything has been changed.
	bool animate(bool faraway, float time, int crack, u32 daynight_ratio);

	scene::SMesh* getMesh()
	{
		return m_mesh;
	}

	bool isAnimationForced() const
	{
		return m_animation_force_timer == 0;
	}

	void decreaseAnimationForceTimer()
	{
		if(m_animation_force_timer > 0)
			m_animation_force_timer--;
	}

private:
	scene::SMesh *m_mesh;
	IGameDef *m_gamedef;

	// Must animate() be called before rendering?
	bool m_has_animation;
	int m_animation_force_timer;

	// Animation info: cracks
	// Last crack value passed to animate()
	int m_last_crack;
	// Maps mesh buffer (i.e. material) indices to base texture names
	std::map<u32, std::string> m_crack_materials;

	// Animation info: texture animationi
	// Maps meshbuffers to TileSpecs
	std::map<u32, TileSpec> m_animation_tiles;
	std::map<u32, int> m_animation_frames; // last animation frame
	std::map<u32, int> m_animation_frame_offsets;
	
	// Animation info: day/night transitions
	// Last daynight_ratio value passed to animate()
	u32 m_last_daynight_ratio;
	// For each meshbuffer, maps vertex indices to (day,night) pairs
	std::map<u32, std::map<u32, std::pair<u8, u8> > > m_daynight_diffs;
};



/*
	This is used because CMeshBuffer::append() is very slow
*/
struct PreMeshBuffer
{
	TileSpec tile;
	core::array<u16> indices;
	core::array<video::S3DVertex> vertices;
};

struct MeshCollector
{
	core::array<PreMeshBuffer> prebuffers;

	void append(const TileSpec &material,
			const video::S3DVertex *vertices, u32 numVertices,
			const u16 *indices, u32 numIndices);
};

// This encodes
//   alpha in the A channel of the returned SColor
//   day light (0-255) in the R channel of the returned SColor
//   night light (0-255) in the G channel of the returned SColor
inline video::SColor MapBlock_LightColor(u8 alpha, u16 light)
{
	return video::SColor(alpha, (light & 0xff), (light >> 8), 0);
}

// Compute light at node
u16 getInteriorLight(MapNode n, s32 increment, MeshMakeData *data);
u16 getFaceLight(MapNode n, MapNode n2, v3s16 face_dir, MeshMakeData *data);
u16 getSmoothLight(v3s16 p, v3s16 corner, MeshMakeData *data);

// Retrieves the TileSpec of a face of a node
// Adds MATERIAL_FLAG_CRACK if the node is cracked
TileSpec getNodeTileN(MapNode mn, v3s16 p, u8 tileindex, MeshMakeData *data);
TileSpec getNodeTile(MapNode mn, v3s16 p, v3s16 dir, MeshMakeData *data);

#endif

s="hl kwb">float brightness = brightness_step; for (size_t i = 0; i < LIGHT_MAX; i++) { light_LUT[i] = (u8)(255 * powf(brightness / 255.0f, gamma)); light_LUT[i] = rangelim(light_LUT[i] + adjustments[i], 0, 255); if (i > 1 && light_LUT[i] < light_LUT[i-1]) light_LUT[i] = light_LUT[i-1] + 1; brightness += brightness_step; } light_LUT[LIGHT_MAX] = 255; } #endif #if 0 /* Made using this and: - adding 220 as the second last one - replacing the third last one (212) with 195 #!/usr/bin/python from math import * from sys import stdout # We want 0 at light=0 and 255 at light=LIGHT_MAX LIGHT_MAX = 14 #FACTOR = 0.69 #FACTOR = 0.75 FACTOR = 0.83 START_FROM_ZERO = False L = [] if START_FROM_ZERO: for i in range(1,LIGHT_MAX+1): L.append(int(round(255.0 * FACTOR ** (i-1)))) L.append(0) else: for i in range(1,LIGHT_MAX+1): L.append(int(round(255.0 * FACTOR ** (i-1)))) L.append(255) L.reverse() for i in L: stdout.write(str(i)+",\n") */ u8 light_decode_table[LIGHT_MAX+1] = { 23, 27, 33, 40, 48, 57, 69, 83, 100, 121, 146, 176, 195, 220, 255, }; #endif #if 0 // This is good // a_n+1 = a_n * 0.786 // Length of LIGHT_MAX+1 means LIGHT_MAX is the last value. // LIGHT_SUN is read as LIGHT_MAX from here. u8 light_decode_table[LIGHT_MAX+1] = { 8, 11, 14, 18, 22, 29, 37, 47, 60, 76, 97, 123, 157, 200, 255, }; #endif #if 0 // Use for debugging in dark u8 light_decode_table[LIGHT_MAX+1] = { 58, 64, 72, 80, 88, 98, 109, 121, 135, 150, 167, 185, 206, 229, 255, }; #endif // This is reasonable with classic lighting with a light source /*u8 light_decode_table[LIGHT_MAX+1] = { 2, 3, 4, 6, 9, 13, 18, 25, 32, 35, 45, 57, 69, 79, 255 };*/ // As in minecraft, a_n+1 = a_n * 0.8 // NOTE: This doesn't really work that well because this defines // LIGHT_MAX as dimmer than LIGHT_SUN // NOTE: Uh, this has had 34 left out; forget this. /*u8 light_decode_table[LIGHT_MAX+1] = { 8, 11, 14, 17, 21, 27, 42, 53, 66, 83, 104, 130, 163, 204, 255, };*/ // This was a quick try of more light, manually quickly made /*u8 light_decode_table[LIGHT_MAX+1] = { 0, 7, 11, 15, 21, 29, 42, 53, 69, 85, 109, 135, 167, 205, 255, };*/ // This was used for a long time, manually made /*u8 light_decode_table[LIGHT_MAX+1] = { 0, 6, 8, 11, 14, 19, 26, 34, 45, 61, 81, 108, 143, 191, 255, };*/ /*u8 light_decode_table[LIGHT_MAX+1] = { 0, 3, 6, 10, 18, 25, 35, 50, 75, 95, 120, 150, 185, 215, 255, };*/ /*u8 light_decode_table[LIGHT_MAX+1] = { 0, 5, 12, 22, 35, 50, 65, 85, 100, 120, 140, 160, 185, 215, 255, };*/ // LIGHT_MAX is 14, 0-14 is 15 values /*u8 light_decode_table[LIGHT_MAX+1] = { 0, 9, 12, 14, 16, 20, 26, 34, 45, 61, 81, 108, 143, 191, 255, };*/ #if 0 /* #!/usr/bin/python from math import * from sys import stdout # We want 0 at light=0 and 255 at light=LIGHT_MAX LIGHT_MAX = 14 #FACTOR = 0.69 FACTOR = 0.75 L = [] for i in range(1,LIGHT_MAX+1): L.append(int(round(255.0 * FACTOR ** (i-1)))) L.append(0) L.reverse() for i in L: stdout.write(str(i)+",\n") */ u8 light_decode_table[LIGHT_MAX+1] = { 0, 6, 8, 11, 14, 19, 26, 34, 45, 61, 81, 108, 143, 191, 255, }; #endif