#include "TGLLightSet.h"
#include "TGLBoundingBox.h"
#include "TGLOrthoCamera.h"
#include "TGLIncludes.h"
ClassImp(TGLLightSet)
TGLLightSet::TGLLightSet() :
TObject(),
fLightState(kLightMask),
fUseSpecular(kTRUE),
fFrontPower(0.4), fSidePower(0.7), fSpecularPower(0.8)
{
}
void TGLLightSet::ToggleLight(ELight light)
{
if (light == kLightSpecular) {
fUseSpecular = !fUseSpecular;
} else if (light >= kLightMask) {
Error("TGLLightSet::ToggleLight", "invalid light type");
return;
} else {
fLightState ^= light;
}
}
void TGLLightSet::SetLight(ELight light, Bool_t on)
{
if (light == kLightSpecular) {
fUseSpecular = on;
} else if (light >= kLightMask) {
Error("TGLViewer::ToggleLight", "invalid light type");
return;
}
if (on) {
fLightState |= light;
} else {
fLightState &= ~light;
}
}
void TGLLightSet::StdSetupLights(const TGLBoundingBox& bbox,
const TGLCamera & camera, Bool_t debug)
{
glPushMatrix();
if (!bbox.IsEmpty())
{
Double_t lightRadius = bbox.Extents().Mag() * 2.9;
Double_t sideLightsZ, frontLightZ;
const TGLOrthoCamera* orthoCamera = dynamic_cast<const TGLOrthoCamera*>(&camera);
if (orthoCamera) {
sideLightsZ =
camera.FrustumPlane(TGLCamera::kNear).DistanceTo(camera.FrustumCenter())*0.7;
frontLightZ = sideLightsZ;
} else {
TGLVector3 eyeVector = camera.EyePoint() - camera.FrustumCenter();
sideLightsZ = eyeVector.Mag() * -0.85;
frontLightZ = 0.2 * lightRadius;
}
glLoadIdentity();
TGLVertex3 c = bbox.Center();
TGLVector3 center(c.X(), c.Y(), c.Z());
camera.RefModelViewMatrix().MultiplyIP(center);
Float_t pos0[] = { 0.0, 0.0, Float_t(frontLightZ), 1.0 };
Float_t pos1[] = { Float_t(center.X()), Float_t(center.Y() + lightRadius), Float_t(sideLightsZ), 1.0 };
Float_t pos2[] = { Float_t(center.X()), Float_t(center.Y() - lightRadius), Float_t(sideLightsZ), 1.0 };
Float_t pos3[] = { Float_t(center.X() - lightRadius), Float_t(center.Y()), Float_t(sideLightsZ), 1.0 };
Float_t pos4[] = { Float_t(center.X() + lightRadius), Float_t(center.Y()), Float_t(sideLightsZ), 1.0 };
Float_t specular = fUseSpecular ? fSpecularPower : 0.0f;
const Float_t frontLightColor[] = { fFrontPower, fFrontPower, fFrontPower, 1.0f };
const Float_t sideLightColor[] = { fSidePower, fSidePower, fSidePower, 1.0f };
const Float_t specLightColor[] = { specular, specular, specular, 1.0f };
glLightfv(GL_LIGHT0, GL_POSITION, pos0);
glLightfv(GL_LIGHT0, GL_DIFFUSE, frontLightColor);
glLightfv(GL_LIGHT0, GL_SPECULAR, specLightColor);
glLightfv(GL_LIGHT1, GL_POSITION, pos1);
glLightfv(GL_LIGHT1, GL_DIFFUSE, sideLightColor);
glLightfv(GL_LIGHT2, GL_POSITION, pos2);
glLightfv(GL_LIGHT2, GL_DIFFUSE, sideLightColor);
glLightfv(GL_LIGHT3, GL_POSITION, pos3);
glLightfv(GL_LIGHT3, GL_DIFFUSE, sideLightColor);
glLightfv(GL_LIGHT4, GL_POSITION, pos4);
glLightfv(GL_LIGHT4, GL_DIFFUSE, sideLightColor);
}
for (UInt_t light = 0; (1<<light) < kLightMask; light++)
{
if ((1<<light) & fLightState)
{
glEnable(GLenum(GL_LIGHT0 + light));
if (debug)
{
glDisable(GL_LIGHTING);
Float_t position[4];
glGetLightfv(GLenum(GL_LIGHT0 + light), GL_POSITION, position);
Double_t size = bbox.Extents().Mag() / 10.0;
TGLVertex3 dPosition(position[0], position[1], position[2]);
TGLUtil::DrawSphere(dPosition, size, TGLUtil::fgYellow);
glEnable(GL_LIGHTING);
}
}
else
{
glDisable(GLenum(GL_LIGHT0 + light));
}
}
glPopMatrix();
}