-/* $Id$
-
-This file is part of libmspgl
-Copyright © 2007 Mikko Rasa, Mikkosoft Productions
-Distributed under the LGPL
-*/
-
-#include "except.h"
+#include <stdexcept>
+#include <msp/strings/format.h>
#include "light.h"
+#include "matrix.h"
#include "misc.h"
+#include "programdata.h"
using namespace std;
namespace GL {
Light::Light():
- ambient(0),
diffuse(1),
specular(1),
position(0, 0, 1, 0),
spot_dir(0, 0, -1),
spot_exp(0),
- spot_cutoff(180)
+ spot_cutoff(Geometry::Angle<float>::straight())
{
- attenuation[0]=1;
- attenuation[1]=0;
- attenuation[2]=0;
+ attenuation[0] = 1;
+ attenuation[1] = 0;
+ attenuation[2] = 0;
}
-void Light::set_ambient(const Color &c)
+void Light::update_matrix()
{
- ambient=c;
+ Vector3 up_dir;
+ if(20*abs(direction.z)>abs(direction.x)+abs(direction.y))
+ up_dir.y = 1;
+ else
+ up_dir.z = 1;
+ Vector3 right_dir = normalize(cross(direction, up_dir));
+
+ Vector4 columns[4];
+ columns[0] = compose(right_dir, 0.0f);
+ columns[1] = compose(cross(right_dir, direction), 0.0f);
+ columns[2] = compose(-direction, 0.0f);
+ columns[3] = position;
+ matrix = Matrix::from_columns(columns);
}
void Light::set_diffuse(const Color &c)
{
- diffuse=c;
+ diffuse = c;
}
void Light::set_specular(const Color &c)
{
- specular=c;
+ specular = c;
+}
+
+void Light::set_matrix(const Matrix &m)
+{
+ Placeable::set_matrix(m);
+ position = matrix.column(3);
+ spot_dir = normalize(-matrix.column(2).slice<3>(0));
+ direction = (position.w ? spot_dir : normalize(-position.slice<3>(0)));
+ update_matrix();
}
void Light::set_position(const Vector4 &p)
{
- position=p;
+ position = p;
+ if(!position.w)
+ direction = normalize(-position.slice<3>(0));
+ update_matrix();
}
void Light::set_spot_direction(const Vector3 &d)
{
- spot_dir=d;
+ spot_dir = normalize(d);
+ if(position.w)
+ direction = spot_dir;
+ update_matrix();
}
void Light::set_spot_exponent(float e)
{
- spot_exp=e;
+ if(e<0)
+ throw invalid_argument("Light::set_spot_exponent");
+
+ spot_exp = e;
}
-void Light::set_spot_cutoff(float c)
+void Light::set_spot_cutoff(const Geometry::Angle<float> &c)
{
- spot_cutoff=c;
+ if(c<Geometry::Angle<float>::zero() || (c>Geometry::Angle<float>::right() && c!=Geometry::Angle<float>::straight()))
+ throw invalid_argument("Light::set_spot_cutoff");
+
+ spot_cutoff = c;
+}
+
+void Light::disable_spot_cutoff()
+{
+ set_spot_cutoff(Geometry::Angle<float>::straight());
}
void Light::set_attenuation(float c, float l, float q)
{
- attenuation[0]=c;
- attenuation[1]=l;
- attenuation[2]=q;
+ attenuation[0] = c;
+ attenuation[1] = l;
+ attenuation[2] = q;
+}
+
+void Light::update_shader_data(ProgramData &shdata, const Matrix &view_matrix, unsigned i) const
+{
+ string base = format("light_sources[%d]", i);
+ shdata.uniform(base+".position", view_matrix*position);
+ shdata.uniform(base+".diffuse", diffuse);
+ shdata.uniform(base+".specular", specular);
}
-void Light::bind() const
+
+Light::Loader::Loader(Light &l):
+ DataFile::ObjectLoader<Light>(l)
{
- if(current_lights[current_unit]!=this)
- {
- GLenum l=GL_LIGHT0+current_unit;
- enable(l);
- glLightfv(l, GL_AMBIENT, &ambient.r);
- glLightfv(l, GL_DIFFUSE, &diffuse.r);
- glLightfv(l, GL_SPECULAR, &specular.r);
- glLightfv(l, GL_POSITION, &position.x);
- glLightfv(l, GL_SPOT_DIRECTION, &spot_dir.x);
- glLightf(l, GL_SPOT_EXPONENT, spot_exp);
- glLightf(l, GL_SPOT_CUTOFF, spot_cutoff);
- glLightf(l, GL_CONSTANT_ATTENUATION, attenuation[0]);
- glLightf(l, GL_LINEAR_ATTENUATION, attenuation[1]);
- glLightf(l, GL_QUADRATIC_ATTENUATION, attenuation[2]);
- current_lights[current_unit]=this;
- }
+ add("attenuation", &Loader::attenuation);
+ add("diffuse", &Loader::diffuse);
+ add("position", &Loader::position);
+ add("specular", &Loader::specular);
+ add("spot_direction", &Loader::spot_direction);
+ add("spot_exponent", &Loader::spot_exponent);
+ add("spot_cutoff", &Loader::spot_cutoff);
}
-void Light::bind_to(unsigned i) const
+void Light::Loader::attenuation(float c, float l, float q)
{
- activate(i);
- bind();
+ obj.set_attenuation(c, l, q);
}
-void Light::activate(unsigned i)
+void Light::Loader::diffuse(float r, float g, float b)
{
- static unsigned max_lights=get_i(GL_MAX_LIGHTS);
+ obj.set_diffuse(Color(r, g, b));
+}
- if(i>=max_lights)
- throw InvalidParameterValue("Light unit index out of range");
+void Light::Loader::position(float x, float y, float z, float w)
+{
+ obj.set_position(Vector4(x, y, z, w));
+}
- if(i>=current_lights.size())
- current_lights.resize(i+1);
+void Light::Loader::specular(float r, float g, float b)
+{
+ obj.set_specular(Color(r, g, b));
+}
- current_unit=i;
+void Light::Loader::spot_direction(float x, float y, float z)
+{
+ obj.set_spot_direction(Vector3(x, y, z));
}
-void Light::unbind()
+void Light::Loader::spot_exponent(float e)
{
- if(current_lights[current_unit])
- {
- disable(GL_LIGHT0+current_unit);
- current_lights[current_unit]=0;
- }
+ obj.set_spot_exponent(e);
}
-unsigned Light::current_unit=0;
-vector<const Light *> Light::current_lights(1);
+void Light::Loader::spot_cutoff(float c)
+{
+ obj.set_spot_cutoff(Geometry::Angle<float>::from_degrees(c));
+}
} // namespace GL
} // namespace Msp