Saxum/game/level.cc

347 lines
9.5 KiB
C++

#include "level.hh"
#include "loader.hh"
#include <string>
Level::Level(std::string xmlFilePath){
// default value
skydomeSize = 50.0f;
physics = Physics();
this->xmlFilePath = xmlFilePath;
}
Level::Level() {
}
Level::~Level() {
if (luaState!=nullptr) {
lua_close(luaState);
}
for(unsigned int i = 0; i<objects.size(); i++) {
delete(objects.at(i));
}
delete(waterPlane);
}
void Level::load() {
//Intialize lua state
// Check if there's an existing state and close it
if(luaState != nullptr){
lua_close(luaState);
luaState = nullptr;
}
// Create a new lua state
luaState = luaL_newstate();
luaL_openlibs(luaState);
//Expose the class Level and its functions to Lua
luabridge::getGlobalNamespace(luaState)
.beginClass<Level>("Level")
.addFunction("deleteObject", &Level::deleteObject)
.addFunction("getObjectCount", &Level::getPhysicsObjectsVectorSize)
.addFunction("moveObject", &Level::moveObject)
.addFunction("resetPlayer", &Level::resetPlayer)
.addFunction("movePlayer", &Level::movePlayer)
.addFunction("setSunDirection", &Level::setSunDirection)
.addFunction("forceMove", &Level::forceMove)
.addFunction("activateEndgame", &Level::activateEndgame)
.addFunction("preloadLightPosition", &Level::preloadLightPosition)
.addFunction("addLightByParameters", &Level::addLightByParameters)
.addFunction("deleteFourLights", &Level::deleteFourLights)
.endClass();
//Push the level to Lua as a global variable
luabridge::push(luaState, this);
lua_setglobal(luaState, "level");
this->camera = Camera(glm::vec2(-0.8f, 0.0f), 3.0f);
}
void Level::render(ACGL::OpenGL::SharedShaderProgram shader, bool lightingPass,
glm::mat4* viewProjectionMatrix, std::vector<glm::mat4>* shadowVPs) {
for(unsigned int i = 0; i<objects.size(); i++) {
if (lightingPass) {
objects.at(i)->render(shader, lightingPass, true, viewProjectionMatrix, shadowVPs);
}
else {
objects.at(i)->render(shader, lightingPass, false, viewProjectionMatrix, shadowVPs);
}
}
if (lightingPass && waterPlane) {
waterPlane->render(shader, lightingPass, true, viewProjectionMatrix, shadowVPs);
}
}
void Level::update(float runTimeSinceLastUpdate, float runTime, glm::vec2 mouseDelta,
KeyboardState* keyboardState) {
// Ignore first two mouse updates, because they are incorrect
// DON'T try to move this functionallity elsewhere
static int i = 0;
if (i <20) {
i++;
mouseDelta.x=mouseDelta.y=0;
}
int runs = 4;
if(i>=20)
{
mouseDelta.x = -mouseDelta.x;
}
for(int j = runs; j > 0; j--)
{
physics.takeUpdateStep(runTimeSinceLastUpdate/runs);
camera.updateRotation(mouseDelta/100.0f/(float)runs);
physics.updateCameraPos(mouseDelta, 50/runs, camera.getDistance());
camera.setPosition(physics.getCameraPosition());
camera.setDirection(physics.getCameraToPlayer());
if(keyboardState->wPressed){
physics.rollForward(camera.getVector(),strength/runs);
}
if(keyboardState->aPressed) {
physics.rollLeft(camera.getVector(),strength/runs);
}
if(keyboardState->sPressed) {
physics.rollBack(camera.getVector(),strength/runs);
}
if(keyboardState->dPressed){
physics.rollRight(camera.getVector(),strength/runs);
}
}
if(keyboardState->f1Pressed) {
physics.forcePlayer(glm::vec3(17.5, 21.0, 87.0));
}
if(keyboardState->f2Pressed) {
physics.forcePlayer(glm::vec3(-78.5, 21.75, 4.5));
}
if(keyboardState->f3Pressed) {
physics.forcePlayer(glm::vec3(-169.5, 21.5, 58.5));
}
if(keyboardState->f4Pressed) {
physics.forcePlayer(glm::vec3(-180.5, 21.75, 58.5));
}
if(keyboardState->kPressed)
camera.setIsPhysicsCamera(true);
if(keyboardState->lPressed)
camera.setIsPhysicsCamera(false);
cameraCenter->setPosition(physics.getPos(0));
cameraCenter->setRotation(physics.getRotation(0));
for(unsigned i = 0; i < physicsObjects.size();i++)
{
physicsObjects[i]->setPosition(physics.getPos(i));
physicsObjects[i]->setRotation(physics.getRotation(i));
}
skydome.setPosition(glm::vec3(cameraCenter->getPosition().x,
0.0f, cameraCenter->getPosition().z));
if (runTime > 2.0f) {
for(unsigned int i = 0; i<triggers.size(); i++) {
triggers.at(i).triggerUpdate();
}
}
}
glm::vec3 Level::getAmbientLight() {
return ambientLight;
}
std::vector<Light>* Level::getLights() {
return &lights;
}
Camera* Level::getCamera() {
return &camera;
}
Object* Level::getCameraCenter() {
return cameraCenter;
}
Light* Level::getDirectionalLight() {
return &directionalLight;
}
glm::vec4 Level::getFogColourDay() {
return fogColourDay;
}
glm::vec4 Level::getFogColourRise() {
return fogColourRise;
}
glm::vec4 Level::getFogColourNight() {
return fogColourNight;
}
glm::vec3 Level::getCameraPosition() {
return cameraCenter->getPosition() + camera.getVector();
}
void Level::setSkydomeSize(float size) {
skydomeSize = size;
}
float Level::getSkydomeSize() {
return this->skydomeSize;
}
std::vector<Object*>* Level::getObjects() {
return &objects;
}
std::vector<Object*>* Level::getPhysicsObjects() {
return &physicsObjects;
}
void Level::moveObject(int objectIndex, float strength, float xPos, float yPos, float zPos){
glm::vec3 position = glm::vec3(xPos, yPos, zPos);
physics.removePositionConstraint(objectIndex);
physics.addPositionConstraint(objectIndex, strength, position);
}
//should not be used since objects does not get synchronized and deletion is not implemented in pyhsics
void Level::deleteObject(int objectIndex){
physicsObjects.erase(physicsObjects.begin() + objectIndex);
for(unsigned int i = 0; i<triggers.size(); i++) {
if(triggers.at(i).deleteNotification(objectIndex)){
triggers.erase(triggers.begin() + i);
i--;
}
}
}
void Level::resetPlayer(){
Loader loader = Loader();
glm::vec3 newPosition = loader.reloadPlayerPosition(xmlFilePath, this);
physics.forcePlayer(newPosition);
}
void Level::movePlayer(float xPosition, float yPosition, float zPosition){
glm::vec3 newPosition = glm::vec3(xPosition, yPosition, zPosition);
physics.forcePlayer(newPosition);
}
void Level::setPlayerIndex(int index){
playerIndex = index;
}
void Level::setStrength(float strength) {
this->strength = strength;
}
void Level::setSkydomeObject(Skydome object){
this->skydome = object;
}
void Level::addObject(Object* object) {
objects.push_back(object);
}
void Level::addPhysicsObject(Object* object) {
physicsObjects.push_back(object);
}
void Level::setAmbientLight(glm::vec3 colour) {
this->ambientLight = colour;
}
void Level::setFogColourDay(glm::vec4 colour) {
this->fogColourDay = colour;
}
void Level::setFogColourRise(glm::vec4 colour) {
this->fogColourRise = colour;
}
void Level::setFogColourNight(glm::vec4 colour) {
this->fogColourNight = colour;
}
void Level::setDirectionalLight(Light light) {
this->directionalLight = light;
}
void Level::setSunDirection(float x, float y, float z){
glm::vec3 lightPosition = glm::vec3(x,y,z);
glm::vec3 lightColour = this->directionalLight.getColour();
float lightIntensity = this->directionalLight.getIntensity();
this->directionalLight = Light(lightPosition, lightColour, lightIntensity);
}
Physics* Level::getPhysics() {
return &physics;
}
unsigned int Level::getObjectsVectorSize() {
return objects.size();
}
unsigned int Level::getPhysicsObjectsVectorSize() {
return physicsObjects.size();
}
void Level::setCameraCenter(Object* object) {
this->cameraCenter = object;
}
void Level::addLight(Light light) {
this->lights.push_back(light);
}
void Level::preloadLightPosition(float xPos, float yPos, float zPos){
nextLightPosition = glm::vec3(xPos, yPos, zPos);
}
void Level::addLightByParameters(float redColour, float greenColour, float blueColour, float intensity, float flameYOffset, float flameHeight, float flameWidth){
glm::vec3 colour = glm::vec3(redColour, greenColour, blueColour);
this->lights.push_back(Light(nextLightPosition, colour, intensity, flameYOffset, flameHeight, flameWidth));
}
void Level::deleteFourLights(){
int indice = lights.size()-4;
lights.erase(lights.begin() + indice);
lights.erase(lights.begin() + indice);
lights.erase(lights.begin() + indice);
lights.erase(lights.begin() + indice);
}
void Level::addTrigger(Trigger trigger) {
this->triggers.push_back(trigger);
}
lua_State* Level::getLuaState() {
return luaState;
}
Terrain* Level::getTerrain() {
return &terrain;
}
Skydome* Level::getSkydome() {
return &skydome;
}
void Level::forceMove(float x, float y, float z, unsigned indice){
glm::vec3 position = glm::vec3(x,y,z);
physics.forceMove(position, indice);
}
void Level::activateEndgame(){
physics.activateEndgame();
}
void Level::setWaterPlane(Object* water) {
this->waterPlane = water;
}
void Level::setTerrain(Terrain terrain) {
this->terrain = terrain;
}