lovr/src/modules/graphics/material.c

122 lines
4.0 KiB
C

#include "graphics/material.h"
#include "graphics/graphics.h"
#include "graphics/shader.h"
#include "graphics/texture.h"
#include "resources/shaders.h"
#include "core/util.h"
#include <stdlib.h>
#include <math.h>
struct Material {
uint32_t ref;
float scalars[MAX_MATERIAL_SCALARS];
Color colors[MAX_MATERIAL_COLORS];
struct Texture* textures[MAX_MATERIAL_TEXTURES];
float transform[9];
};
Material* lovrMaterialCreate() {
Material* material = calloc(1, sizeof(Material));
lovrAssert(material, "Out of memory");
material->ref = 1;
for (int i = 0; i < MAX_MATERIAL_SCALARS; i++) {
material->scalars[i] = i == SCALAR_ALPHA_CUTOFF ? 0.f : 1.f;
}
for (int i = 0; i < MAX_MATERIAL_COLORS; i++) {
if (i == COLOR_EMISSIVE) {
material->colors[i] = (Color) { 0.f, 0.f, 0.f, 0.f };
} else {
material->colors[i] = (Color) { 1.f, 1.f, 1.f, 1.f };
}
}
lovrMaterialSetTransform(material, 0.f, 0.f, 1.f, 1.f, 0.f);
return material;
}
void lovrMaterialDestroy(void* ref) {
Material* material = ref;
lovrGraphicsFlushMaterial(material);
for (int i = 0; i < MAX_MATERIAL_TEXTURES; i++) {
lovrRelease(material->textures[i], lovrTextureDestroy);
}
free(material);
}
void lovrMaterialBind(Material* material, Shader* shader) {
for (int i = 0; i < MAX_MATERIAL_SCALARS; i++) {
lovrShaderSetFloats(shader, lovrShaderScalarUniforms[i], &material->scalars[i], 0, 1);
}
for (int i = 0; i < MAX_MATERIAL_COLORS; i++) {
lovrShaderSetColor(shader, lovrShaderColorUniforms[i], material->colors[i]);
}
for (int i = 0; i < MAX_MATERIAL_TEXTURES; i++) {
lovrShaderSetTextures(shader, lovrShaderTextureUniforms[i], &material->textures[i], 0, 1);
}
lovrShaderSetMatrices(shader, "lovrMaterialTransform", material->transform, 0, 9);
}
float lovrMaterialGetScalar(Material* material, MaterialScalar scalarType) {
return material->scalars[scalarType];
}
void lovrMaterialSetScalar(Material* material, MaterialScalar scalarType, float value) {
if (material->scalars[scalarType] != value) {
lovrGraphicsFlushMaterial(material);
material->scalars[scalarType] = value;
}
}
Color lovrMaterialGetColor(Material* material, MaterialColor colorType) {
return material->colors[colorType];
}
void lovrMaterialSetColor(Material* material, MaterialColor colorType, Color color) {
if (memcmp(&material->colors[colorType], &color, 4 * sizeof(float))) {
lovrGraphicsFlushMaterial(material);
material->colors[colorType] = color;
}
}
Texture* lovrMaterialGetTexture(Material* material, MaterialTexture textureType) {
return material->textures[textureType];
}
void lovrMaterialSetTexture(Material* material, MaterialTexture textureType, Texture* texture) {
if (material->textures[textureType] != texture) {
lovrAssert(!texture || lovrTextureGetType(texture) == TEXTURE_2D, "Material textures must be 2D");
lovrGraphicsFlushMaterial(material);
lovrRetain(texture);
lovrRelease(material->textures[textureType], lovrTextureDestroy);
material->textures[textureType] = texture;
}
}
void lovrMaterialGetTransform(Material* material, float* ox, float* oy, float* sx, float* sy, float* angle) {
*ox = material->transform[6];
*oy = material->transform[7];
*sx = sqrtf(material->transform[0] * material->transform[0] + material->transform[1] * material->transform[1]);
*sy = sqrtf(material->transform[3] * material->transform[3] + material->transform[4] * material->transform[4]);
*angle = atan2f(-material->transform[3], material->transform[0]);
}
void lovrMaterialSetTransform(Material* material, float ox, float oy, float sx, float sy, float angle) {
lovrGraphicsFlushMaterial(material);
float c = cosf(angle);
float s = sinf(angle);
material->transform[0] = c * sx;
material->transform[1] = s * sx;
material->transform[2] = 0.f;
material->transform[3] = -s * sy;
material->transform[4] = c * sy;
material->transform[5] = 0.f;
material->transform[6] = ox;
material->transform[7] = oy;
material->transform[8] = 1.f;
}