-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathmaterial.h
137 lines (121 loc) · 5.84 KB
/
material.h
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
// Copyright (C) 2020-2021 Sami Väisänen
// Copyright (C) 2020-2021 Ensisoft http://www.ensisoft.com
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 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 General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
#pragma once
#include "config.h"
#include <string>
#include <variant>
#include <vector>
#include <unordered_map>
#include "base/assert.h"
#include "graphics/enum.h"
#include "graphics/types.h"
#include "graphics/device.h"
namespace gfx
{
class Device;
class ShaderSource;
class TextureSource;
class MaterialClass;
// Material instance. Each instance can contain and alter the
// instance specific state even between instances that refer
// to the same underlying material type.
class Material
{
public:
using Flags = MaterialFlags;
using Uniform = gfx::Uniform;
using UniformMap = gfx::UniformMap;
using RenderPass = gfx::RenderPass;
struct Environment {
// true if running in an "editing mode", which means that even
// content marked static might have changed and should be checked
// in case it has been modified and should be re-uploaded.
bool editing_mode = false;
DrawPrimitive draw_primitive = DrawPrimitive::Triangles;
DrawCategory draw_category = DrawCategory::Basic;
RenderPass renderpass = RenderPass::ColorPass;
};
struct RasterState {
using Blending = Device::State::BlendOp;
Blending blending = Blending::None;
bool premultiplied_alpha = false;
};
using RuntimeValue = std::variant<double, float, int, std::string>;
using CommandArg = std::variant<float, int, std::string>;
struct Command {
std::string name;
std::unordered_map<std::string, CommandArg> args;
};
using CommandList = std::vector<Command>;
virtual ~Material() = default;
virtual void SetFlag(Flags flag, bool on_off) noexcept { }
virtual bool TestFlag(Flags flag) const noexcept { return false; }
// Apply the dynamic material properties to the given program object
// and set the rasterizer state. Dynamic properties are the properties
// that can change between one material instance to another even when
// the underlying material type is the same. For example two instances
// of material "marble" have the same underlying static type "marble"
// but both instances have their own instance state as well.
// Returns true if the state is applied correctly and drawing can proceed.
virtual bool ApplyDynamicState(const Environment& env, Device& device, ProgramState& program, RasterState& raster) const = 0;
// Apply the static state, i.e. the material state that doesn't change
// during the material's lifetime and need to be only set once.
virtual void ApplyStaticState(const Environment& env, Device& device, ProgramState& program) const = 0;
// Get the device specific shader source applicable for this material, its state
// and the given environment in which it should execute.
virtual ShaderSource GetShader(const Environment& env, const Device& device) const = 0;
// Get the shader ID applicable for this material, its state and the given
// environment in which it should execute.
virtual std::string GetShaderId(const Environment& env) const = 0;
// Get the human readable debug name that should be associated with the
// shader object generated from this drawable.
virtual std::string GetShaderName(const Environment& env) const = 0;
// Get the material class id (if any).
virtual std::string GetClassId() const { return {}; }
// Update material time by a delta value (in seconds).
virtual void Update(float dt) {}
// Set the material instance time to a specific time value.
virtual void SetRuntime(double runtime) {}
// Set material instance uniform.
virtual void SetUniform(const std::string& name, Uniform value) {}
// Set all material uniforms at once.
virtual void SetUniforms(UniformMap uniforms) {}
// Clear away all material instance uniforms.
virtual void ResetUniforms() {}
// Get the current material time.
virtual double GetRuntime() const { return 0.0; }
// Execute a material command to change the material in some way
// or for example run a sprite cycle animation.
virtual bool Execute(const Environment& env, const Command& command) { return true; }
virtual bool GetValue(const std::string& key, RuntimeValue* value) const { return false; }
// Get the material class instance if any. Warning, this may be null for
// material objects that aren't based on any material clas!
virtual const MaterialClass* GetClass() const { return nullptr; }
template<typename T>
inline bool GetValue(const std::string& key, T* out) const noexcept
{
RuntimeValue value;
if (GetValue(key, &value))
{
ASSERT(std::holds_alternative<T>(value));
*out = std::get<T>(value);
return true;
}
return false;
}
private:
};
} // namespace