1487 lines
46 KiB
C++
1487 lines
46 KiB
C++
#include "renderer_light.hpp"
|
|
|
|
#include <functional>
|
|
#include <math.h>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "tinythread.h"
|
|
|
|
#include "LuaTools.h"
|
|
|
|
#include "modules/Gui.h"
|
|
#include "modules/Maps.h"
|
|
#include "modules/Screen.h"
|
|
#include "modules/Units.h"
|
|
|
|
#include "df/block_square_event_material_spatterst.h"
|
|
#include "df/building.h"
|
|
#include "df/building_doorst.h"
|
|
#include "df/building_floodgatest.h"
|
|
#include "df/flow_info.h"
|
|
#include "df/graphic.h"
|
|
#include "df/item.h"
|
|
#include "df/items_other_id.h"
|
|
#include "df/plant.h"
|
|
#include "df/plant_raw.h"
|
|
#include "df/unit.h"
|
|
#include "df/viewscreen_dungeonmodest.h"
|
|
#include "df/viewscreen_dwarfmodest.h"
|
|
#include "df/world.h"
|
|
|
|
using df::global::gps;
|
|
using namespace DFHack;
|
|
using df::coord2d;
|
|
using namespace tthread;
|
|
|
|
const float RootTwo = 1.4142135623730950488016887242097f;
|
|
|
|
|
|
bool isInRect(const coord2d& pos,const rect2d& rect)
|
|
{
|
|
if(pos.x>=rect.first.x && pos.y>=rect.first.y && pos.x<rect.second.x && pos.y<rect.second.y)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
lightingEngineViewscreen::~lightingEngineViewscreen()
|
|
{
|
|
threading.shutdown();
|
|
}
|
|
|
|
lightSource::lightSource(rgbf power,int radius):power(power),flicker(false)
|
|
{
|
|
if(radius >= 0)
|
|
this->radius = radius;
|
|
else
|
|
{
|
|
float levelDim = 0.2f;//TODO this is not correct if you change config
|
|
float totalPower = power.r;
|
|
if(totalPower < power.g)totalPower = power.g;
|
|
if(totalPower < power.b)totalPower = power.b;
|
|
if(totalPower > 0 && levelDim > 0)
|
|
this->radius = (int)((log(levelDim/totalPower)/log(0.85f))) + 1;
|
|
else
|
|
this->radius = 0;
|
|
}
|
|
}
|
|
|
|
rect2d getMapViewport()
|
|
{
|
|
const int AREA_MAP_WIDTH = 23;
|
|
const int MENU_WIDTH = 30;
|
|
if(!gps || !df::viewscreen_dwarfmodest::_identity.is_instance(DFHack::Gui::getCurViewscreen()))
|
|
{
|
|
if(gps && df::viewscreen_dungeonmodest::_identity.is_instance(DFHack::Gui::getCurViewscreen()))
|
|
{
|
|
return mkrect_wh(0,0,gps->dimx,gps->dimy);
|
|
}
|
|
else
|
|
return mkrect_wh(0,0,0,0);
|
|
|
|
}
|
|
int w=gps->dimx;
|
|
int h=gps->dimy;
|
|
int view_height=h-2;
|
|
int area_x2 = w-AREA_MAP_WIDTH-2;
|
|
int menu_x2=w-MENU_WIDTH-2;
|
|
int menu_x1=area_x2-MENU_WIDTH-1;
|
|
int view_rb=w-1;
|
|
|
|
int area_pos=(*df::global::ui_menu_width)[1];
|
|
int menu_pos=(*df::global::ui_menu_width)[0];
|
|
if(area_pos<3)
|
|
{
|
|
view_rb=area_x2;
|
|
}
|
|
if (menu_pos<area_pos || df::global::ui->main.mode!=0)
|
|
{
|
|
if (menu_pos >= area_pos)
|
|
menu_pos = area_pos-1;
|
|
int menu_x = menu_x2;
|
|
if(menu_pos < 2) menu_x = menu_x1;
|
|
view_rb = menu_x;
|
|
}
|
|
return mkrect_wh(1,1,view_rb,view_height+1);
|
|
}
|
|
lightingEngineViewscreen::lightingEngineViewscreen(renderer_light* target):lightingEngine(target),threading(this),doDebug(false)
|
|
{
|
|
reinit();
|
|
defaultSettings();
|
|
int numTreads=tthread::thread::hardware_concurrency();
|
|
if(numTreads==0)numTreads=1;
|
|
threading.start(numTreads);
|
|
}
|
|
|
|
void lightingEngineViewscreen::reinit()
|
|
{
|
|
if(!gps)
|
|
return;
|
|
w=gps->dimx;
|
|
h=gps->dimy;
|
|
size_t size=w*h;
|
|
lightMap.resize(size,rgbf(1,1,1));
|
|
ocupancy.resize(size);
|
|
lights.resize(size);
|
|
}
|
|
|
|
void plotCircle(int xm, int ym, int r,const std::function<void(int,int)>& setPixel)
|
|
{
|
|
int x = -r, y = 0, err = 2-2*r; /* II. Quadrant */
|
|
do {
|
|
setPixel(xm-x, ym+y); /* I. Quadrant */
|
|
setPixel(xm-y, ym-x); /* II. Quadrant */
|
|
setPixel(xm+x, ym-y); /* III. Quadrant */
|
|
setPixel(xm+y, ym+x); /* IV. Quadrant */
|
|
r = err;
|
|
if (r <= y) err += ++y*2+1; /* e_xy+e_y < 0 */
|
|
if (r > x || err > y) err += ++x*2+1; /* e_xy+e_x > 0 or no 2nd y-step */
|
|
} while (x < 0);
|
|
}
|
|
void plotSquare(int xm, int ym, int r,const std::function<void(int,int)>& setPixel)
|
|
{
|
|
for(int x = 0; x <= r; x++)
|
|
{
|
|
setPixel(xm+r, ym+x); /* I.1 Quadrant */
|
|
setPixel(xm+x, ym+r); /* I.2 Quadrant */
|
|
setPixel(xm+r, ym-x); /* II.1 Quadrant */
|
|
setPixel(xm+x, ym-r); /* II.2 Quadrant */
|
|
setPixel(xm-r, ym-x); /* III.1 Quadrant */
|
|
setPixel(xm-x, ym-r); /* III.2 Quadrant */
|
|
setPixel(xm-r, ym+x); /* IV.1 Quadrant */
|
|
setPixel(xm-x, ym+r); /* IV.2 Quadrant */
|
|
}
|
|
}
|
|
void plotLine(int x0, int y0, int x1, int y1,rgbf power,const std::function<rgbf(rgbf,int,int,int,int)>& setPixel)
|
|
{
|
|
int dx = abs(x1-x0), sx = x0<x1 ? 1 : -1;
|
|
int dy = -abs(y1-y0), sy = y0<y1 ? 1 : -1;
|
|
int err = dx+dy, e2; /* error value e_xy */
|
|
int rdx=0;
|
|
int rdy=0;
|
|
for(;;){ /* loop */
|
|
if(rdx!=0 || rdy!=0) //dirty hack to skip occlusion on the first tile.
|
|
{
|
|
power=setPixel(power,rdx,rdy,x0,y0);
|
|
if(power.dot(power)<0.00001f)
|
|
return ;
|
|
}
|
|
if (x0==x1 && y0==y1) break;
|
|
e2 = 2*err;
|
|
rdx=rdy=0;
|
|
if (e2 >= dy) { err += dy; x0 += sx; rdx=sx;} /* e_xy+e_x > 0 */
|
|
if (e2 <= dx) { err += dx; y0 += sy; rdy=sy;} /* e_xy+e_y < 0 */
|
|
}
|
|
return ;
|
|
}
|
|
void plotLineDiffuse(int x0, int y0, int x1, int y1,rgbf power,int num_diffuse,const std::function<rgbf(rgbf,int,int,int,int)>& setPixel,bool skip_hack=false)
|
|
{
|
|
|
|
int dx = abs(x1-x0), sx = x0<x1 ? 1 : -1;
|
|
int dy = -abs(y1-y0), sy = y0<y1 ? 1 : -1;
|
|
int dsq=dx*dx+dy*dy;
|
|
int err = dx+dy, e2; /* error value e_xy */
|
|
int rdx=0;
|
|
int rdy=0;
|
|
for(;;){ /* loop */
|
|
if(rdx!=0 || rdy!=0 || skip_hack) //dirty hack to skip occlusion on the first tile.
|
|
{
|
|
power=setPixel(power,rdx,rdy,x0,y0);
|
|
if(power.dot(power)<0.00001f)
|
|
return ;
|
|
}
|
|
if (x0==x1 && y0==y1) break;
|
|
e2 = 2*err;
|
|
rdx=rdy=0;
|
|
if (e2 >= dy) { err += dy; x0 += sx; rdx=sx;} /* e_xy+e_x > 0 */
|
|
if (e2 <= dx) { err += dx; y0 += sy; rdy=sy;} /* e_xy+e_y < 0 */
|
|
|
|
if(num_diffuse>0 && dsq/4<(x1-x0)*(x1-x0)+(y1-y0)*(y1-y0))//reached center?
|
|
{
|
|
const float betta=0.25;
|
|
int nx=y1-y0; //right angle
|
|
int ny=x1-x0;
|
|
if((nx*nx+ny*ny)*betta*betta>2)
|
|
{
|
|
plotLineDiffuse(x0,y0,x0+nx*betta,y0+ny*betta,power,num_diffuse-1,setPixel,true);
|
|
plotLineDiffuse(x0,y0,x0-nx*betta,y0-ny*betta,power,num_diffuse-1,setPixel,true);
|
|
}
|
|
}
|
|
}
|
|
return ;
|
|
}
|
|
void plotLineAA(int x0, int y0, int x1, int y1,rgbf power,const std::function<rgbf(rgbf,int,int,int,int)>& setPixelAA)
|
|
{
|
|
int dx = abs(x1-x0), sx = x0<x1 ? 1 : -1;
|
|
int dy = abs(y1-y0), sy = y0<y1 ? 1 : -1;
|
|
int err = dx-dy, e2, x2; /* error value e_xy */
|
|
int ed = dx+dy == 0 ? 1 : sqrt((float)dx*dx+(float)dy*dy);
|
|
int rdx=0;
|
|
int rdy=0;
|
|
int lrdx,lrdy;
|
|
rgbf sumPower;
|
|
for ( ; ; ){ /* pixel loop */
|
|
float strsum=0;
|
|
float str=1-abs(err-dx+dy)/(float)ed;
|
|
strsum=str;
|
|
sumPower=setPixelAA(power*str,rdx,rdy,x0,y0);
|
|
e2 = err; x2 = x0;
|
|
lrdx=rdx;
|
|
lrdy=rdy;
|
|
rdx=rdy=0;
|
|
if (2*e2 >= -dx) { /* x step */
|
|
if (x0 == x1) break;
|
|
|
|
if (e2+dy < ed)
|
|
{
|
|
str=1-(e2+dy)/(float)ed;
|
|
sumPower+=setPixelAA(power*str,lrdx,lrdy,x0,y0+sy);
|
|
strsum+=str;
|
|
}
|
|
err -= dy; x0 += sx; rdx=sx;
|
|
}
|
|
if (2*e2 <= dy) { /* y step */
|
|
if (y0 == y1) break;
|
|
|
|
if (dx-e2 < ed)
|
|
{
|
|
str=1-(dx-e2)/(float)ed;
|
|
sumPower+=setPixelAA(power*str,lrdx,lrdy,x2+sx,y0);
|
|
strsum+=str;
|
|
}
|
|
err += dx; y0 += sy; rdy=sy;
|
|
}
|
|
if(strsum<0.001f)
|
|
return;
|
|
sumPower=sumPower/strsum;
|
|
if(sumPower.dot(sumPower)<0.00001f)
|
|
return;
|
|
power=sumPower;
|
|
}
|
|
}
|
|
rgbf blendMax(const rgbf& a,const rgbf& b)
|
|
{
|
|
return rgbf(std::max(a.r,b.r),std::max(a.g,b.g),std::max(a.b,b.b));
|
|
}
|
|
rgbf blend(const rgbf& a,const rgbf& b)
|
|
{
|
|
return blendMax(a,b);
|
|
}
|
|
void lightingEngineViewscreen::clear()
|
|
{
|
|
lightMap.assign(lightMap.size(),rgbf(1,1,1));
|
|
tthread::lock_guard<tthread::fast_mutex> guard(myRenderer->dataMutex);
|
|
if(lightMap.size()==myRenderer->lightGrid.size())
|
|
{
|
|
std::swap(myRenderer->lightGrid,lightMap);
|
|
myRenderer->invalidate();
|
|
}
|
|
}
|
|
void lightingEngineViewscreen::calculate()
|
|
{
|
|
if(lightMap.size()!=myRenderer->lightGrid.size())
|
|
{
|
|
reinit();
|
|
myRenderer->invalidate();//needs a lock?
|
|
}
|
|
rect2d vp=getMapViewport();
|
|
const rgbf dim(levelDim,levelDim,levelDim);
|
|
lightMap.assign(lightMap.size(),rgbf(1,1,1));
|
|
lights.assign(lights.size(),lightSource());
|
|
for(int i=vp.first.x;i<vp.second.x;i++)
|
|
for(int j=vp.first.y;j<vp.second.y;j++)
|
|
{
|
|
lightMap[getIndex(i,j)]=dim;
|
|
}
|
|
doOcupancyAndLights();
|
|
threading.signalDoneOcclusion();
|
|
threading.waitForWrites();
|
|
}
|
|
void lightingEngineViewscreen::updateWindow()
|
|
{
|
|
tthread::lock_guard<tthread::fast_mutex> guard(myRenderer->dataMutex);
|
|
if(lightMap.size()!=myRenderer->lightGrid.size())
|
|
{
|
|
reinit();
|
|
myRenderer->invalidate();
|
|
return;
|
|
}
|
|
|
|
bool isAdventure=(*df::global::gametype==df::game_type::ADVENTURE_ARENA)||
|
|
(*df::global::gametype==df::game_type::ADVENTURE_MAIN);
|
|
if(isAdventure)
|
|
{
|
|
fixAdvMode(adv_mode);
|
|
}
|
|
|
|
if(doDebug)
|
|
std::swap(ocupancy,myRenderer->lightGrid);
|
|
else
|
|
std::swap(lightMap,myRenderer->lightGrid);
|
|
rect2d vp=getMapViewport();
|
|
|
|
myRenderer->invalidateRect(vp.first.x,vp.first.y,vp.second.x-vp.first.x,vp.second.y-vp.first.y);
|
|
}
|
|
void lightingEngineViewscreen::preRender()
|
|
{
|
|
|
|
}
|
|
void lightingEngineViewscreen::fixAdvMode(int mode)
|
|
{
|
|
|
|
MapExtras::MapCache mc;
|
|
const rgbf dim(levelDim,levelDim,levelDim);
|
|
rect2d vp=getMapViewport();
|
|
int window_x=*df::global::window_x;
|
|
int window_y=*df::global::window_y;
|
|
int window_z=*df::global::window_z;
|
|
//mode 0-> make dark non-visible parts
|
|
if(mode==0)
|
|
{
|
|
for(int x=vp.first.x;x<vp.second.x;x++)
|
|
for(int y=vp.first.y;y<vp.second.y;y++)
|
|
{
|
|
df::tile_designation d=mc.designationAt(DFCoord(window_x+x,window_y+y,window_z));
|
|
if(d.bits.pile!=1)
|
|
{
|
|
lightMap[getIndex(x,y)]=dim;
|
|
}
|
|
}
|
|
}
|
|
//mode 1-> make everything visible, let the lighting hide stuff
|
|
else if(mode==1)
|
|
{
|
|
for(int x=vp.first.x;x<vp.second.x;x++)
|
|
for(int y=vp.first.y;y<vp.second.y;y++)
|
|
{
|
|
df::tile_designation d=mc.designationAt(DFCoord(window_x+x,window_y+y,window_z));
|
|
d.bits.dig=df::tile_dig_designation::Default;//TODO add union and flag there
|
|
d.bits.hidden=0;
|
|
d.bits.pile = 1;
|
|
mc.setDesignationAt(DFCoord(window_x+x,window_y+y,window_z),d);
|
|
}
|
|
}
|
|
}
|
|
void lightSource::combine(const lightSource& other)
|
|
{
|
|
power=blend(power,other.power);
|
|
radius=std::max(other.radius,radius);//hack... but who cares
|
|
}
|
|
bool lightingEngineViewscreen::addLight(int tileId,const lightSource& light)
|
|
{
|
|
bool wasLight=lights[tileId].radius>0;
|
|
lights[tileId].combine(light);
|
|
if(light.flicker)
|
|
lights[tileId].flicker=true;
|
|
return wasLight;
|
|
}
|
|
void lightingEngineViewscreen::addOclusion(int tileId,const rgbf& c,float thickness)
|
|
{
|
|
if(thickness > 0.999 && thickness < 1.001)
|
|
ocupancy[tileId]*=c;
|
|
else
|
|
ocupancy[tileId]*=(c.pow(thickness));
|
|
}
|
|
rgbf getStandartColor(int colorId)
|
|
{
|
|
return rgbf(df::global::enabler->ccolor[colorId][0]/255.0f,
|
|
df::global::enabler->ccolor[colorId][1]/255.0f,
|
|
df::global::enabler->ccolor[colorId][2]/255.0f);
|
|
}
|
|
int getPlantNumber(const std::string& id)
|
|
{
|
|
std::vector<df::plant_raw*>& vec=df::plant_raw::get_vector();
|
|
for(size_t i=0;i<vec.size();i++)
|
|
{
|
|
if(vec[i]->id==id)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
void addPlant(const std::string& id,std::map<int,lightSource>& map,const lightSource& v)
|
|
{
|
|
int nId=getPlantNumber(id);
|
|
if(nId>0)
|
|
{
|
|
map[nId]=v;
|
|
}
|
|
}
|
|
matLightDef* lightingEngineViewscreen::getMaterialDef( int matType,int matIndex )
|
|
{
|
|
auto it=matDefs.find(std::make_pair(matType,matIndex));
|
|
if(it!=matDefs.end())
|
|
return &it->second;
|
|
else
|
|
return NULL;
|
|
}
|
|
buildingLightDef* lightingEngineViewscreen::getBuildingDef( df::building* bld )
|
|
{
|
|
auto it=buildingDefs.find(std::make_tuple((int)bld->getType(),(int)bld->getSubtype(),(int)bld->getCustomType()));
|
|
if(it!=buildingDefs.end())
|
|
return &it->second;
|
|
else
|
|
return NULL;
|
|
}
|
|
creatureLightDef* lightingEngineViewscreen::getCreatureDef(df::unit* u)
|
|
{
|
|
auto it=creatureDefs.find(std::make_pair(int(u->race),int(u->caste)));
|
|
if(it!=creatureDefs.end())
|
|
return &it->second;
|
|
else
|
|
{
|
|
auto it2=creatureDefs.find(std::make_pair(int(u->race),int(-1)));
|
|
if(it2!=creatureDefs.end())
|
|
return &it2->second;
|
|
else
|
|
return NULL;
|
|
}
|
|
}
|
|
itemLightDef* lightingEngineViewscreen::getItemDef(df::item* it)
|
|
{
|
|
auto iter=itemDefs.find(std::make_pair(int(it->getType()),int(it->getSubtype())));
|
|
if(iter!=itemDefs.end())
|
|
return &iter->second;
|
|
else
|
|
{
|
|
auto iter2=itemDefs.find(std::make_pair(int(it->getType()),int(-1)));
|
|
if(iter2!=itemDefs.end())
|
|
return &iter2->second;
|
|
else
|
|
return NULL;
|
|
}
|
|
}
|
|
void lightingEngineViewscreen::applyMaterial(int tileId,const matLightDef& mat,float size, float thickness)
|
|
{
|
|
if(mat.isTransparent)
|
|
{
|
|
addOclusion(tileId,mat.transparency,thickness);
|
|
}
|
|
else
|
|
ocupancy[tileId]=rgbf(0,0,0);
|
|
if(mat.isEmiting)
|
|
addLight(tileId,mat.makeSource(size));
|
|
}
|
|
bool lightingEngineViewscreen::applyMaterial(int tileId,int matType,int matIndex,float size,float thickness,const matLightDef* def)
|
|
{
|
|
matLightDef* m=getMaterialDef(matType,matIndex);
|
|
if(m)
|
|
{
|
|
applyMaterial(tileId,*m,size,thickness);
|
|
return true;
|
|
}
|
|
else if(def)
|
|
{
|
|
applyMaterial(tileId,*def,size,thickness);
|
|
}
|
|
return false;
|
|
}
|
|
rgbf lightingEngineViewscreen::propogateSun(MapExtras::Block* b, int x,int y,const rgbf& in,bool lastLevel)
|
|
{
|
|
//TODO unify under addLight/addOclusion
|
|
const rgbf matStairCase(0.9f,0.9f,0.9f);
|
|
rgbf ret=in;
|
|
coord2d innerCoord(x,y);
|
|
df::tiletype type = b->staticTiletypeAt(innerCoord);
|
|
df::tile_designation d = b->DesignationAt(innerCoord);
|
|
//df::tile_occupancy o = b->OccupancyAt(innerCoord);
|
|
df::tiletype_shape shape = ENUM_ATTR(tiletype,shape,type);
|
|
df::tiletype_shape_basic basic_shape = ENUM_ATTR(tiletype_shape, basic_shape, shape);
|
|
DFHack::t_matpair mat=b->staticMaterialAt(innerCoord);
|
|
df::tiletype_material tileMat= ENUM_ATTR(tiletype,material,type);
|
|
|
|
matLightDef* lightDef;
|
|
if(tileMat==df::tiletype_material::FROZEN_LIQUID)
|
|
{
|
|
df::tiletype typeIce = b->tiletypeAt(innerCoord);
|
|
df::tiletype_shape shapeIce = ENUM_ATTR(tiletype,shape,typeIce);
|
|
df::tiletype_shape_basic basicShapeIce = ENUM_ATTR(tiletype_shape,basic_shape,shapeIce);
|
|
if(basicShapeIce==df::tiletype_shape_basic::Wall)
|
|
ret*=matIce.transparency;
|
|
else if(basicShapeIce==df::tiletype_shape_basic::Floor || basicShapeIce==df::tiletype_shape_basic::Ramp || shapeIce==df::tiletype_shape::STAIR_UP)
|
|
if(!lastLevel)
|
|
ret*=matIce.transparency.pow(1.0f/7.0f);
|
|
}
|
|
|
|
lightDef=getMaterialDef(mat.mat_type,mat.mat_index);
|
|
|
|
if(!lightDef || !lightDef->isTransparent)
|
|
lightDef=&matWall;
|
|
if(basic_shape==df::tiletype_shape_basic::Wall)
|
|
{
|
|
ret*=lightDef->transparency;
|
|
}
|
|
else if(basic_shape==df::tiletype_shape_basic::Floor || basic_shape==df::tiletype_shape_basic::Ramp || shape==df::tiletype_shape::STAIR_UP)
|
|
{
|
|
|
|
if(!lastLevel)
|
|
ret*=lightDef->transparency.pow(1.0f/7.0f);
|
|
}
|
|
else if(shape==df::tiletype_shape::STAIR_DOWN || shape==df::tiletype_shape::STAIR_UPDOWN)
|
|
{
|
|
ret*=matStairCase;
|
|
}
|
|
if(d.bits.liquid_type == df::enums::tile_liquid::Water && d.bits.flow_size > 0)
|
|
{
|
|
ret *=matWater.transparency.pow((float)d.bits.flow_size/7.0f);
|
|
}
|
|
else if(d.bits.liquid_type == df::enums::tile_liquid::Magma && d.bits.flow_size > 0)
|
|
{
|
|
ret *=matLava.transparency.pow((float)d.bits.flow_size/7.0f);
|
|
}
|
|
return ret;
|
|
}
|
|
coord2d lightingEngineViewscreen::worldToViewportCoord(const coord2d& in,const rect2d& r,const coord2d& window2d)
|
|
{
|
|
return in-window2d+r.first;
|
|
}
|
|
|
|
static size_t max_list_size = 100000; // Avoid iterating over huge lists
|
|
void lightingEngineViewscreen::doSun(const lightSource& sky,MapExtras::MapCache& map)
|
|
{
|
|
//TODO fix this mess
|
|
int window_x=*df::global::window_x;
|
|
int window_y=*df::global::window_y;
|
|
coord2d window2d(window_x,window_y);
|
|
int window_z=*df::global::window_z;
|
|
rect2d vp=getMapViewport();
|
|
coord2d vpSize=rect_size(vp);
|
|
rect2d blockVp;
|
|
blockVp.first=window2d/16;
|
|
blockVp.second=(window2d+vpSize)/16;
|
|
blockVp.second.x=std::min(blockVp.second.x,(int16_t)df::global::world->map.x_count_block);
|
|
blockVp.second.y=std::min(blockVp.second.y,(int16_t)df::global::world->map.y_count_block);
|
|
//endof mess
|
|
for(int blockX=blockVp.first.x;blockX<=blockVp.second.x;blockX++)
|
|
for(int blockY=blockVp.first.y;blockY<=blockVp.second.y;blockY++)
|
|
{
|
|
rgbf cellArray[16][16];
|
|
for(int block_x = 0; block_x < 16; block_x++)
|
|
for(int block_y = 0; block_y < 16; block_y++)
|
|
cellArray[block_x][block_y] = sky.power;
|
|
|
|
int emptyCell=0;
|
|
for(int z=window_z;z< df::global::world->map.z_count && emptyCell<256;z++)
|
|
{
|
|
MapExtras::Block* b=map.BlockAt(DFCoord(blockX,blockY,z));
|
|
if(!b)
|
|
continue;
|
|
emptyCell=0;
|
|
for(int block_x = 0; block_x < 16; block_x++)
|
|
for(int block_y = 0; block_y < 16; block_y++)
|
|
{
|
|
rgbf& curCell=cellArray[block_x][block_y];
|
|
curCell=propogateSun(b,block_x,block_y,curCell,z==window_z);
|
|
if(curCell.dot(curCell)<0.003f)
|
|
emptyCell++;
|
|
}
|
|
}
|
|
if(emptyCell==256)
|
|
continue;
|
|
for(int block_x = 0; block_x < 16; block_x++)
|
|
for(int block_y = 0; block_y < 16; block_y++)
|
|
{
|
|
rgbf& curCell=cellArray[block_x][block_y];
|
|
df::coord2d pos;
|
|
pos.x = blockX*16+block_x;
|
|
pos.y = blockY*16+block_y;
|
|
pos=worldToViewportCoord(pos,vp,window2d);
|
|
if(isInRect(pos,vp) && curCell.dot(curCell)>0.003f)
|
|
{
|
|
lightSource sun=lightSource(curCell,15);
|
|
addLight(getIndex(pos.x,pos.y),sun);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
rgbf lightingEngineViewscreen::getSkyColor(float v)
|
|
{
|
|
if(dayColors.size()<2)
|
|
{
|
|
v=abs(fmod(v+0.5,1)-0.5)*2;
|
|
return rgbf(v,v,v);
|
|
}
|
|
else
|
|
{
|
|
float pos=v*(dayColors.size()-1);
|
|
int pre=floor(pos);
|
|
pos-=pre;
|
|
if(pre==int(dayColors.size())-1)
|
|
return dayColors[pre];
|
|
return dayColors[pre]*(1-pos)+dayColors[pre+1]*pos;
|
|
}
|
|
}
|
|
void lightingEngineViewscreen::doOcupancyAndLights()
|
|
{
|
|
float daycol;
|
|
if(dayHour<0)
|
|
{
|
|
int length=1200/daySpeed;
|
|
daycol= (*df::global::cur_year_tick % length)/ (float)length;
|
|
}
|
|
else
|
|
daycol= fmod(dayHour,24.0f)/24.0f; //1->12h 0->24h
|
|
|
|
rgbf sky_col=getSkyColor(daycol);
|
|
lightSource sky(sky_col, -1);//auto calculate best size
|
|
|
|
MapExtras::MapCache cache;
|
|
doSun(sky,cache);
|
|
|
|
int window_x=*df::global::window_x;
|
|
int window_y=*df::global::window_y;
|
|
coord2d window2d(window_x,window_y);
|
|
int window_z=*df::global::window_z;
|
|
rect2d vp=getMapViewport();
|
|
coord2d vpSize=rect_size(vp);
|
|
rect2d blockVp;
|
|
blockVp.first=coord2d(window_x,window_y)/16;
|
|
blockVp.second=(window2d+vpSize)/16;
|
|
blockVp.second.x=std::min(blockVp.second.x,(int16_t)df::global::world->map.x_count_block);
|
|
blockVp.second.y=std::min(blockVp.second.y,(int16_t)df::global::world->map.y_count_block);
|
|
|
|
for(int blockX=blockVp.first.x;blockX<=blockVp.second.x;blockX++)
|
|
for(int blockY=blockVp.first.y;blockY<=blockVp.second.y;blockY++)
|
|
{
|
|
MapExtras::Block* b=cache.BlockAt(DFCoord(blockX,blockY,window_z));
|
|
MapExtras::Block* bDown=cache.BlockAt(DFCoord(blockX,blockY,window_z-1));
|
|
if(!b)
|
|
continue; //empty blocks fixed by sun propagation
|
|
|
|
for(int block_x = 0; block_x < 16; block_x++)
|
|
for(int block_y = 0; block_y < 16; block_y++)
|
|
{
|
|
df::coord2d pos;
|
|
pos.x = blockX*16+block_x;
|
|
pos.y = blockY*16+block_y;
|
|
df::coord2d gpos=pos;
|
|
pos=worldToViewportCoord(pos,vp,window2d);
|
|
if(!isInRect(pos,vp))
|
|
continue;
|
|
int tile=getIndex(pos.x,pos.y);
|
|
rgbf& curCell=ocupancy[tile];
|
|
curCell=matAmbience.transparency;
|
|
|
|
|
|
df::tiletype type = b->tiletypeAt(gpos);
|
|
df::tile_designation d = b->DesignationAt(gpos);
|
|
if(d.bits.hidden )
|
|
{
|
|
curCell=rgbf(0,0,0);
|
|
continue; // do not process hidden stuff, TODO other hidden stuff
|
|
}
|
|
//df::tile_occupancy o = b->OccupancyAt(gpos);
|
|
df::tiletype_shape shape = ENUM_ATTR(tiletype,shape,type);
|
|
bool is_wall=!ENUM_ATTR(tiletype_shape,passable_high,shape);
|
|
bool is_floor=!ENUM_ATTR(tiletype_shape,passable_low,shape);
|
|
df::tiletype_shape_basic basic_shape = ENUM_ATTR(tiletype_shape, basic_shape, shape);
|
|
df::tiletype_material tileMat= ENUM_ATTR(tiletype,material,type);
|
|
|
|
DFHack::t_matpair mat=b->staticMaterialAt(gpos);
|
|
|
|
matLightDef* lightDef=getMaterialDef(mat.mat_type,mat.mat_index);
|
|
if(!lightDef || !lightDef->isTransparent)
|
|
lightDef=&matWall;
|
|
if(shape==df::tiletype_shape::BROOK_BED )
|
|
{
|
|
curCell=rgbf(0,0,0);
|
|
}
|
|
else if(is_wall)
|
|
{
|
|
if(tileMat==df::tiletype_material::FROZEN_LIQUID)
|
|
applyMaterial(tile,matIce);
|
|
else
|
|
applyMaterial(tile,*lightDef);
|
|
}
|
|
else if(!d.bits.liquid_type && d.bits.flow_size>0 )
|
|
{
|
|
applyMaterial(tile,matWater, (float)d.bits.flow_size/7.0f, (float)d.bits.flow_size/7.0f);
|
|
}
|
|
if(d.bits.liquid_type && d.bits.flow_size>0)
|
|
{
|
|
applyMaterial(tile,matLava,(float)d.bits.flow_size/7.0f,(float)d.bits.flow_size/7.0f);
|
|
}
|
|
else if(!is_floor)
|
|
{
|
|
if(bDown)
|
|
{
|
|
df::tile_designation d2=bDown->DesignationAt(gpos);
|
|
if(d2.bits.liquid_type && d2.bits.flow_size>0)
|
|
{
|
|
applyMaterial(tile,matLava);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
|
|
df::map_block* block=b->getRaw();
|
|
if(!block)
|
|
continue;
|
|
//flows
|
|
for(size_t i=0;i<block->flows.size();i++)
|
|
{
|
|
df::flow_info* f=block->flows[i];
|
|
if(f && f->density>0 && (f->type==df::flow_type::Dragonfire || f->type==df::flow_type::Fire))
|
|
{
|
|
df::coord2d pos=f->pos;
|
|
pos=worldToViewportCoord(pos,vp,window2d);
|
|
int tile=getIndex(pos.x,pos.y);
|
|
if(isInRect(pos,vp))
|
|
{
|
|
rgbf fireColor;
|
|
if(f->density>60)
|
|
{
|
|
fireColor=rgbf(0.98f,0.91f,0.30f);
|
|
}
|
|
else if(f->density>30)
|
|
{
|
|
fireColor=rgbf(0.93f,0.16f,0.16f);
|
|
}
|
|
else
|
|
{
|
|
fireColor=rgbf(0.64f,0.0f,0.0f);
|
|
}
|
|
lightSource fire(fireColor,f->density/5);
|
|
addLight(tile,fire);
|
|
}
|
|
}
|
|
}
|
|
|
|
//blood and other goo
|
|
for(size_t i=0;i<block->block_events.size();i++)
|
|
{
|
|
df::block_square_event* ev=block->block_events[i];
|
|
df::block_square_event_type ev_type=ev->getType();
|
|
if(ev_type==df::block_square_event_type::material_spatter)
|
|
{
|
|
df::block_square_event_material_spatterst* spatter=static_cast<df::block_square_event_material_spatterst*>(ev);
|
|
matLightDef* m=getMaterialDef(spatter->mat_type,spatter->mat_index);
|
|
if(!m)
|
|
continue;
|
|
if(!m->isEmiting)
|
|
continue;
|
|
for(int x=0;x<16;x++)
|
|
for(int y=0;y<16;y++)
|
|
{
|
|
df::coord2d pos;
|
|
pos.x = blockX*16+x;
|
|
pos.y = blockY*16+y;
|
|
int16_t amount=spatter->amount[x][y];
|
|
if(amount<=0)
|
|
continue;
|
|
pos=worldToViewportCoord(pos,vp,window2d);
|
|
if(isInRect(pos,vp))
|
|
{
|
|
addLight(getIndex(pos.x,pos.y),m->makeSource((float)amount/100));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if(df::global::cursor->x>-30000)
|
|
{
|
|
int wx=df::global::cursor->x-window_x+vp.first.x;
|
|
int wy=df::global::cursor->y-window_y+vp.first.y;
|
|
int tile=getIndex(wx,wy);
|
|
applyMaterial(tile,matCursor);
|
|
}
|
|
//citizen only emit light, if defined
|
|
//or other creatures
|
|
if(matCitizen.isEmiting || creatureDefs.size()>0)
|
|
for (size_t i=0;i<df::global::world->units.active.size();++i)
|
|
{
|
|
df::unit *u = df::global::world->units.active[i];
|
|
coord2d pos=worldToViewportCoord(coord2d(u->pos.x,u->pos.y),vp,window2d);
|
|
if(u->pos.z==window_z && isInRect(pos,vp))
|
|
{
|
|
if (DFHack::Units::isCitizen(u) && !u->counters.unconscious)
|
|
addLight(getIndex(pos.x,pos.y),matCitizen.makeSource());
|
|
creatureLightDef *def=getCreatureDef(u);
|
|
if(def && !u->flags1.bits.dead)
|
|
{
|
|
addLight(getIndex(pos.x,pos.y),def->light.makeSource());
|
|
}
|
|
}
|
|
}
|
|
//items
|
|
if(itemDefs.size()>0)
|
|
{
|
|
std::vector<df::item*>& vec=df::global::world->items.other[items_other_id::IN_PLAY];
|
|
for(size_t i=0;i<vec.size();i++)
|
|
{
|
|
df::item* curItem=vec[i];
|
|
df::coord itemPos=DFHack::Items::getPosition(curItem);
|
|
coord2d pos=worldToViewportCoord(itemPos,vp,window2d);
|
|
itemLightDef* mat=0;
|
|
if( itemPos.z==window_z && isInRect(pos,vp) && (mat=getItemDef(curItem)) )
|
|
{
|
|
if( ((mat->equiped || mat->haul ||mat->inBuilding ||mat->inContainer) && curItem->flags.bits.in_inventory)|| //TODO split this up
|
|
(mat->onGround && curItem->flags.bits.on_ground) )
|
|
{
|
|
if(mat->light.isEmiting)
|
|
addLight(getIndex(pos.x,pos.y),mat->light.makeSource());
|
|
if(!mat->light.isTransparent)
|
|
addOclusion(getIndex(pos.x,pos.y),mat->light.transparency,1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//buildings
|
|
for(size_t i = 0; i < df::global::world->buildings.all.size(); i++)
|
|
{
|
|
df::building *bld = df::global::world->buildings.all[i];
|
|
|
|
if(window_z!=bld->z)
|
|
continue;
|
|
if(bld->getBuildStage()<bld->getMaxBuildStage()) //only work if fully built
|
|
continue;
|
|
|
|
df::coord2d p1(bld->x1,bld->y1);
|
|
df::coord2d p2(bld->x2,bld->y2);
|
|
p1=worldToViewportCoord(p1,vp,window2d);
|
|
p2=worldToViewportCoord(p2,vp,window2d);
|
|
if(isInRect(p1,vp)||isInRect(p2,vp))
|
|
{
|
|
|
|
int tile;
|
|
if(isInRect(p1,vp))
|
|
tile=getIndex(p1.x,p1.y); //TODO multitile buildings. How they would work?
|
|
else
|
|
tile=getIndex(p2.x,p2.y);
|
|
df::building_type type = bld->getType();
|
|
buildingLightDef* def=getBuildingDef(bld);
|
|
if(!def)
|
|
continue;
|
|
if(type==df::enums::building_type::Door)
|
|
{
|
|
df::building_doorst* door=static_cast<df::building_doorst*>(bld);
|
|
if(!door->door_flags.bits.closed)
|
|
continue;
|
|
}
|
|
else if(type==df::enums::building_type::Floodgate)
|
|
{
|
|
df::building_floodgatest* gate=static_cast<df::building_floodgatest*>(bld);
|
|
if(!gate->gate_flags.bits.closed)
|
|
continue;
|
|
}
|
|
|
|
|
|
if(def->useMaterial)
|
|
{
|
|
matLightDef* mat=getMaterialDef(bld->mat_type,bld->mat_index);
|
|
if(!mat)mat=&matWall;
|
|
if(!def->poweredOnly || !bld->isUnpowered()) //not powered. Add occlusion only.
|
|
{
|
|
if(def->light.isEmiting)
|
|
{
|
|
addLight(tile,def->light.makeSource(def->size));
|
|
}
|
|
else if(mat->isEmiting)
|
|
{
|
|
addLight(tile,mat->makeSource(def->size));
|
|
}
|
|
}
|
|
if(def->light.isTransparent)
|
|
{
|
|
addOclusion(tile,def->light.transparency,def->size);
|
|
}
|
|
else
|
|
{
|
|
addOclusion(tile,mat->transparency,def->size);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(!def->poweredOnly || !bld->isUnpowered())//not powered. Add occlusion only.
|
|
addOclusion(tile,def->light.transparency,def->size);
|
|
else
|
|
applyMaterial(tile,def->light,def->size,def->thickness);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
rgbf lua_parseLightCell(lua_State* L)
|
|
{
|
|
rgbf ret;
|
|
|
|
lua_pushnumber(L,1);
|
|
lua_gettable(L,-2);
|
|
ret.r=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
|
|
lua_pushnumber(L,2);
|
|
lua_gettable(L,-2);
|
|
ret.g=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
|
|
lua_pushnumber(L,3);
|
|
lua_gettable(L,-2);
|
|
ret.b=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
|
|
//Lua::GetOutput(L)->print("got cell(%f,%f,%f)\n",ret.r,ret.g,ret.b);
|
|
return ret;
|
|
}
|
|
#define GETLUAFLAG(field,name) lua_getfield(L,-1,"flags");\
|
|
if(lua_isnil(L,-1)){field=false;}\
|
|
else{lua_getfield(L,-1,#name);field=lua_isnil(L,-1);lua_pop(L,1);}\
|
|
lua_pop(L,1)
|
|
|
|
#define GETLUANUMBER(field,name) lua_getfield(L,-1,#name);\
|
|
if(!lua_isnil(L,-1) && lua_isnumber(L,-1))field=lua_tonumber(L,-1);\
|
|
lua_pop(L,1)
|
|
matLightDef lua_parseMatDef(lua_State* L)
|
|
{
|
|
|
|
matLightDef ret;
|
|
lua_getfield(L,-1,"tr");
|
|
if((ret.isTransparent=!lua_isnil(L,-1)))
|
|
{
|
|
ret.transparency=lua_parseLightCell(L);
|
|
}
|
|
lua_pop(L,1);
|
|
|
|
lua_getfield(L,-1,"em");
|
|
if((ret.isEmiting=!lua_isnil(L,-1)))
|
|
{
|
|
ret.emitColor=lua_parseLightCell(L);
|
|
lua_pop(L,1);
|
|
lua_getfield(L,-1,"rad");
|
|
if(lua_isnil(L,-1))
|
|
{
|
|
lua_pop(L,1);
|
|
luaL_error(L,"Material has emittance but no radius");
|
|
}
|
|
else
|
|
ret.radius=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
}
|
|
else
|
|
lua_pop(L,1);
|
|
GETLUAFLAG(ret.flicker,"flicker");
|
|
return ret;
|
|
}
|
|
int lightingEngineViewscreen::parseMaterials(lua_State* L)
|
|
{
|
|
auto engine= (lightingEngineViewscreen*)lua_touserdata(L, 1);
|
|
engine->matDefs.clear();
|
|
//color_ostream* os=Lua::GetOutput(L);
|
|
Lua::StackUnwinder unwinder(L);
|
|
lua_getfield(L,2,"materials");
|
|
if(!lua_istable(L,-1))
|
|
{
|
|
luaL_error(L,"Materials table not found.");
|
|
return 0;
|
|
}
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
int type=lua_tonumber(L,-2);
|
|
//os->print("Processing type:%d\n",type);
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
int index=lua_tonumber(L,-2);
|
|
//os->print("\tProcessing index:%d\n",index);
|
|
engine->matDefs[std::make_pair(type,index)]=lua_parseMatDef(L);
|
|
|
|
lua_pop(L, 1);
|
|
}
|
|
lua_pop(L, 1);
|
|
}
|
|
return 0;
|
|
}
|
|
#define LOAD_SPECIAL(lua_name,class_name) \
|
|
lua_getfield(L,-1,#lua_name);\
|
|
if(!lua_isnil(L,-1))engine->class_name=lua_parseMatDef(L);\
|
|
lua_pop(L,1)
|
|
int lightingEngineViewscreen::parseSpecial(lua_State* L)
|
|
{
|
|
auto engine= (lightingEngineViewscreen*)lua_touserdata(L, 1);
|
|
Lua::StackUnwinder unwinder(L);
|
|
lua_getfield(L,2,"special");
|
|
if(!lua_istable(L,-1))
|
|
{
|
|
luaL_error(L,"Special table not found.");
|
|
return 0;
|
|
}
|
|
LOAD_SPECIAL(LAVA,matLava);
|
|
LOAD_SPECIAL(WATER,matWater);
|
|
LOAD_SPECIAL(FROZEN_LIQUID,matIce);
|
|
LOAD_SPECIAL(AMBIENT,matAmbience);
|
|
LOAD_SPECIAL(CURSOR,matCursor);
|
|
LOAD_SPECIAL(CITIZEN,matCitizen);
|
|
GETLUANUMBER(engine->levelDim,levelDim);
|
|
GETLUANUMBER(engine->dayHour,dayHour);
|
|
GETLUANUMBER(engine->daySpeed,daySpeed);
|
|
GETLUANUMBER(engine->num_diffuse,diffusionCount);
|
|
GETLUANUMBER(engine->adv_mode,advMode);
|
|
lua_getfield(L,-1,"dayColors");
|
|
if(lua_istable(L,-1))
|
|
{
|
|
engine->dayColors.clear();
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
engine->dayColors.push_back(lua_parseLightCell(L));
|
|
lua_pop(L,1);
|
|
}
|
|
lua_pop(L,1);
|
|
}
|
|
return 0;
|
|
}
|
|
#undef LOAD_SPECIAL
|
|
int lightingEngineViewscreen::parseItems(lua_State* L)
|
|
{
|
|
auto engine= (lightingEngineViewscreen*)lua_touserdata(L, 1);
|
|
engine->itemDefs.clear();
|
|
Lua::StackUnwinder unwinder(L);
|
|
lua_getfield(L,2,"items");
|
|
if(!lua_istable(L,-1))
|
|
{
|
|
luaL_error(L,"Items table not found.");
|
|
return 0;
|
|
}
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
if(!lua_istable(L,-1))
|
|
luaL_error(L,"Broken item definitions.");
|
|
lua_getfield(L,-1,"type");
|
|
int type=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
lua_getfield(L,-1,"subtype");
|
|
int subtype=luaL_optinteger(L,-1,-1);
|
|
lua_pop(L,1);
|
|
itemLightDef item;
|
|
lua_getfield(L,-1,"light");
|
|
item.light=lua_parseMatDef(L);
|
|
GETLUAFLAG(item.haul,"hauling");
|
|
GETLUAFLAG(item.equiped,"equiped");
|
|
GETLUAFLAG(item.inBuilding,"inBuilding");
|
|
GETLUAFLAG(item.inContainer,"contained");
|
|
GETLUAFLAG(item.onGround,"onGround");
|
|
GETLUAFLAG(item.useMaterial,"useMaterial");
|
|
engine->itemDefs[std::make_pair(type,subtype)]=item;
|
|
lua_pop(L,2);
|
|
}
|
|
lua_pop(L,1);
|
|
return 0;
|
|
}
|
|
int lightingEngineViewscreen::parseCreatures(lua_State* L)
|
|
{
|
|
auto engine= (lightingEngineViewscreen*)lua_touserdata(L, 1);
|
|
engine->creatureDefs.clear();
|
|
Lua::StackUnwinder unwinder(L);
|
|
lua_getfield(L,2,"creatures");
|
|
if(!lua_istable(L,-1))
|
|
{
|
|
luaL_error(L,"Creatures table not found.");
|
|
return 0;
|
|
}
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
if(!lua_istable(L,-1))
|
|
luaL_error(L,"Broken creature definitions.");
|
|
lua_getfield(L,-1,"race");
|
|
int race=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
lua_getfield(L,-1,"caste");
|
|
int caste=lua_tonumber(L,-1);
|
|
lua_pop(L,1);
|
|
creatureLightDef cr;
|
|
lua_getfield(L,-1,"light");
|
|
cr.light=lua_parseMatDef(L);
|
|
engine->creatureDefs[std::make_pair(race,caste)]=cr;
|
|
lua_pop(L,2);
|
|
}
|
|
lua_pop(L,1);
|
|
return 0;
|
|
}
|
|
int lightingEngineViewscreen::parseBuildings(lua_State* L)
|
|
{
|
|
auto engine= (lightingEngineViewscreen*)lua_touserdata(L, 1);
|
|
engine->buildingDefs.clear();
|
|
Lua::StackUnwinder unwinder(L);
|
|
lua_getfield(L,2,"buildings");
|
|
if(!lua_istable(L,-1))
|
|
{
|
|
luaL_error(L,"Buildings table not found.");
|
|
return 0;
|
|
}
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
int type=lua_tonumber(L,-2);
|
|
if(!lua_istable(L,-1))
|
|
{
|
|
luaL_error(L,"Broken building definitions.");
|
|
}
|
|
//os->print("Processing type:%d\n",type);
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
int subtype=lua_tonumber(L,-2);
|
|
//os->print("\tProcessing subtype:%d\n",index);
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
int custom=lua_tonumber(L,-2);
|
|
//os->print("\tProcessing custom:%d\n",index);
|
|
buildingLightDef current;
|
|
current.light=lua_parseMatDef(L);
|
|
engine->buildingDefs[std::make_tuple(type,subtype,custom)]=current;
|
|
GETLUAFLAG(current.poweredOnly,"poweredOnly");
|
|
GETLUAFLAG(current.useMaterial,"useMaterial");
|
|
|
|
lua_getfield(L,-1,"size");
|
|
current.size=luaL_optnumber(L,-1,1);
|
|
lua_pop(L,1);
|
|
|
|
lua_getfield(L,-1,"thickness");
|
|
current.thickness=luaL_optnumber(L,-1,1);
|
|
lua_pop(L,1);
|
|
|
|
lua_pop(L, 1);
|
|
}
|
|
|
|
lua_pop(L, 1);
|
|
}
|
|
lua_pop(L, 1);
|
|
}
|
|
lua_pop(L,1);
|
|
return 0;
|
|
}
|
|
void lightingEngineViewscreen::defaultSettings()
|
|
{
|
|
matAmbience=matLightDef(rgbf(0.85f,0.85f,0.85f));
|
|
matLava=matLightDef(rgbf(0.8f,0.2f,0.2f),rgbf(0.8f,0.2f,0.2f),5);
|
|
matWater=matLightDef(rgbf(0.6f,0.6f,0.8f));
|
|
matIce=matLightDef(rgbf(0.7f,0.7f,0.9f));
|
|
matCursor=matLightDef(rgbf(0.96f,0.84f,0.03f),11);
|
|
matCursor.flicker=true;
|
|
matWall=matLightDef(rgbf(0,0,0));
|
|
matCitizen=matLightDef(rgbf(0.8f,0.8f,0.9f),6);
|
|
levelDim=0.2f;
|
|
dayHour=-1;
|
|
daySpeed=1;
|
|
adv_mode=0;
|
|
num_diffuse=0;
|
|
dayColors.push_back(rgbf(0,0,0));
|
|
dayColors.push_back(rgbf(1,1,1));
|
|
dayColors.push_back(rgbf(0,0,0));
|
|
}
|
|
void lightingEngineViewscreen::loadSettings()
|
|
{
|
|
std::string rawFolder;
|
|
if(df::global::world->cur_savegame.save_dir!="")
|
|
{
|
|
rawFolder= "data/save/" + (df::global::world->cur_savegame.save_dir) + "/raw/";
|
|
}
|
|
else
|
|
{
|
|
rawFolder= "raw/";
|
|
}
|
|
const std::string settingsfile=rawFolder+"rendermax.lua";
|
|
|
|
CoreSuspender lock;
|
|
color_ostream_proxy out(Core::getInstance().getConsole());
|
|
|
|
lua_State* s=DFHack::Lua::Core::State;
|
|
lua_newtable(s);
|
|
int env=lua_gettop(s);
|
|
try{
|
|
int ret=luaL_loadfile(s,settingsfile.c_str());
|
|
if(ret==LUA_ERRFILE)
|
|
{
|
|
out.printerr("File not found:%s\n",settingsfile.c_str());
|
|
lua_pop(s,1);
|
|
}
|
|
else if(ret==LUA_ERRSYNTAX)
|
|
{
|
|
out.printerr("Syntax error:\n\t%s\n",lua_tostring(s,-1));
|
|
}
|
|
else
|
|
{
|
|
lua_pushvalue(s,env);
|
|
|
|
if(Lua::SafeCall(out,s,1,0))
|
|
{
|
|
lua_pushcfunction(s, parseMaterials);
|
|
lua_pushlightuserdata(s, this);
|
|
lua_pushvalue(s,env);
|
|
Lua::SafeCall(out,s,2,0);
|
|
out.print("%zu materials loaded\n",matDefs.size());
|
|
|
|
lua_pushcfunction(s, parseSpecial);
|
|
lua_pushlightuserdata(s, this);
|
|
lua_pushvalue(s,env);
|
|
Lua::SafeCall(out,s,2,0);
|
|
out.print("%zu day light colors loaded\n",dayColors.size());
|
|
|
|
lua_pushcfunction(s, parseBuildings);
|
|
lua_pushlightuserdata(s, this);
|
|
lua_pushvalue(s,env);
|
|
Lua::SafeCall(out,s,2,0);
|
|
out.print("%zu buildings loaded\n",buildingDefs.size());
|
|
|
|
lua_pushcfunction(s, parseCreatures);
|
|
lua_pushlightuserdata(s, this);
|
|
lua_pushvalue(s,env);
|
|
Lua::SafeCall(out,s,2,0);
|
|
out.print("%zu creatures loaded\n",creatureDefs.size());
|
|
|
|
lua_pushcfunction(s, parseItems);
|
|
lua_pushlightuserdata(s, this);
|
|
lua_pushvalue(s,env);
|
|
Lua::SafeCall(out,s,2,0);
|
|
out.print("%zu items loaded\n",itemDefs.size());
|
|
}
|
|
|
|
}
|
|
}
|
|
catch(std::exception& e)
|
|
{
|
|
out.printerr("%s",e.what());
|
|
}
|
|
lua_pop(s,1);
|
|
}
|
|
#undef GETLUAFLAG
|
|
#undef GETLUANUMBER
|
|
/*
|
|
* Threading stuff
|
|
*/
|
|
lightThread::lightThread( lightThreadDispatch& dispatch ):dispatch(dispatch),myThread(0),isDone(false)
|
|
{
|
|
|
|
}
|
|
lightThread::~lightThread()
|
|
{
|
|
delete myThread;
|
|
}
|
|
|
|
void lightThread::run()
|
|
{
|
|
while(!isDone)
|
|
{
|
|
//TODO: get area to process, and then process (by rounds): 1. occlusions, 2.sun, 3.lights(could be difficult, units/items etc...)
|
|
{//wait for occlusion (and lights) to be ready
|
|
tthread::lock_guard<tthread::mutex> guard(dispatch.occlusionMutex);
|
|
if(!dispatch.occlusionReady)
|
|
dispatch.occlusionDone.wait(dispatch.occlusionMutex);//wait for work
|
|
if(dispatch.unprocessed.size()==0 || !dispatch.occlusionReady) //spurious wake-up
|
|
continue;
|
|
if(dispatch.occlusion.size()!=canvas.size()) //oh no somebody resized stuff
|
|
canvas.resize(dispatch.occlusion.size());
|
|
}
|
|
|
|
|
|
{ //get my rectangle (any will do)
|
|
tthread::lock_guard<tthread::mutex> guard(dispatch.unprocessedMutex);
|
|
if (dispatch.unprocessed.size()==0)
|
|
{
|
|
//wtf?? why?!
|
|
continue;
|
|
}
|
|
myRect=dispatch.unprocessed.top();
|
|
dispatch.unprocessed.pop();
|
|
if (dispatch.unprocessed.size()==0)
|
|
{
|
|
dispatch.occlusionReady=false;
|
|
}
|
|
|
|
}
|
|
work();
|
|
{
|
|
tthread::lock_guard<tthread::mutex> guard(dispatch.writeLock);
|
|
combine();//write it back
|
|
dispatch.writeCount++;
|
|
}
|
|
dispatch.writesDone.notify_one();//tell about it to the dispatch.
|
|
}
|
|
}
|
|
|
|
void lightThread::work()
|
|
{
|
|
canvas.assign(canvas.size(),rgbf(0,0,0));
|
|
for(int i=myRect.first.x;i<myRect.second.x;i++)
|
|
for(int j=myRect.first.y;j<myRect.second.y;j++)
|
|
{
|
|
doLight(i,j);
|
|
}
|
|
}
|
|
|
|
void lightThread::combine()
|
|
{
|
|
for(size_t i=0;i<canvas.size();i++)
|
|
{
|
|
rgbf& c=dispatch.lightMap[i];
|
|
c=blend(c,canvas[i]);
|
|
}
|
|
}
|
|
|
|
|
|
rgbf lightThread::lightUpCell(rgbf power,int dx,int dy,int tx,int ty)
|
|
{
|
|
int h=dispatch.getH();
|
|
if(isInRect(coord2d(tx,ty),dispatch.viewPort))
|
|
{
|
|
size_t tile=tx*h+ty;
|
|
int dsq=dx*dx+dy*dy;
|
|
float dt=1;
|
|
if(dsq == 1)//array lookup might be faster still
|
|
dt=1;
|
|
else if(dsq == 2)
|
|
dt = RootTwo;
|
|
else if(dsq == 0)
|
|
dt = 0;
|
|
else
|
|
dt=sqrt((float)dsq);
|
|
rgbf& v=dispatch.occlusion[tile];
|
|
lightSource& ls=dispatch.lights[tile];
|
|
bool wallhack=false;
|
|
if(v.r+v.g+v.b==0)
|
|
wallhack=true;
|
|
|
|
if (dsq>0 && !wallhack)
|
|
{
|
|
power*=v.pow(dt);
|
|
}
|
|
if(ls.radius>0 && dsq>0)
|
|
{
|
|
if(power<=ls.power) //quit early if hitting another (stronger) lightsource
|
|
return rgbf();
|
|
}
|
|
|
|
rgbf oldCol=canvas[tile];
|
|
rgbf ncol=blendMax(power,oldCol);
|
|
canvas[tile]=ncol;
|
|
|
|
if(wallhack)
|
|
return rgbf();
|
|
|
|
|
|
return power;
|
|
}
|
|
else
|
|
return rgbf();
|
|
}
|
|
void lightThread::doRay(const rgbf& power,int cx,int cy,int tx,int ty,int num_diffuse)
|
|
{
|
|
using namespace std::placeholders;
|
|
plotLineDiffuse(cx,cy,tx,ty,power,num_diffuse,std::bind(&lightThread::lightUpCell,this,_1,_2,_3,_4,_5));
|
|
}
|
|
|
|
void lightThread::doLight( int x,int y )
|
|
{
|
|
using namespace std::placeholders;
|
|
lightSource& csource=dispatch.lights[x*dispatch.getH()+y];
|
|
int num_diffuse=dispatch.num_diffusion;
|
|
if(csource.radius>0)
|
|
{
|
|
rgbf power=csource.power;
|
|
int radius =csource.radius;
|
|
if(csource.flicker)
|
|
{
|
|
float flicker=(rand()/(float)RAND_MAX)/2.0f+0.5f;
|
|
radius*=flicker;
|
|
power=power*flicker;
|
|
}
|
|
rgbf surrounds;
|
|
lightUpCell( power, 0, 0,x, y); //light up the source itself
|
|
for(int i=-1;i<2;i++)
|
|
for(int j=-1;j<2;j++)
|
|
if(i!=0||j!=0)
|
|
surrounds += lightUpCell( power, i, j,x+i, y+j); //and this is wall hack (so that walls look nice)
|
|
if(surrounds.dot(surrounds)>0.00001f) //if we needed to light up the suroundings, then raycast
|
|
{
|
|
|
|
plotSquare(x,y,radius,
|
|
std::bind(&lightThread::doRay,this,power,x,y,_1,_2,num_diffuse));
|
|
}
|
|
}
|
|
}
|
|
void lightThreadDispatch::signalDoneOcclusion()
|
|
{
|
|
{
|
|
tthread::lock_guard<tthread::mutex> guardWrite(writeLock);
|
|
writeCount=0;
|
|
}
|
|
tthread::lock_guard<tthread::mutex> guard1(occlusionMutex);
|
|
tthread::lock_guard<tthread::mutex> guard2(unprocessedMutex);
|
|
while(!unprocessed.empty())
|
|
unprocessed.pop();
|
|
viewPort=getMapViewport();
|
|
int threadCount=threadPool.size();
|
|
int w=viewPort.second.x-viewPort.first.x;
|
|
int slicew=w/threadCount;
|
|
for(int i=0;i<threadCount;i++)
|
|
{
|
|
rect2d area=viewPort;
|
|
area.first.x=viewPort.first.x+i*slicew;
|
|
if(i==threadCount-1)
|
|
area.second.x=viewPort.second.x; //fix, when area is even number or sth.
|
|
else
|
|
area.second.x=viewPort.first.x+(i+1)*slicew;
|
|
unprocessed.push(area);
|
|
}
|
|
occlusionReady=true;
|
|
occlusionDone.notify_all();
|
|
}
|
|
|
|
lightThreadDispatch::lightThreadDispatch( lightingEngineViewscreen* p ):parent(p),lights(parent->lights),
|
|
occlusionReady(false),occlusion(parent->ocupancy),num_diffusion(parent->num_diffuse),
|
|
lightMap(parent->lightMap),writeCount(0)
|
|
{
|
|
|
|
}
|
|
|
|
void lightThreadDispatch::shutdown()
|
|
{
|
|
for(size_t i=0;i<threadPool.size();i++)
|
|
{
|
|
threadPool[i]->isDone=true;
|
|
|
|
}
|
|
occlusionDone.notify_all();//if stuck signal that you are done with stuff.
|
|
for(size_t i=0;i<threadPool.size();i++)
|
|
{
|
|
threadPool[i]->myThread->join();
|
|
}
|
|
threadPool.clear();
|
|
}
|
|
|
|
int lightThreadDispatch::getW()
|
|
{
|
|
return parent->getW();
|
|
}
|
|
|
|
int lightThreadDispatch::getH()
|
|
{
|
|
return parent->getH();
|
|
}
|
|
void threadStub(void * arg)
|
|
{
|
|
if(arg)
|
|
((lightThread*)arg)->run();
|
|
}
|
|
void lightThreadDispatch::start(int count)
|
|
{
|
|
for(int i=0;i<count;i++)
|
|
{
|
|
std::unique_ptr<lightThread> nthread(new lightThread(*this));
|
|
nthread->myThread=new tthread::thread(&threadStub,nthread.get());
|
|
threadPool.push_back(std::move(nthread));
|
|
}
|
|
}
|
|
|
|
void lightThreadDispatch::waitForWrites()
|
|
{
|
|
tthread::lock_guard<tthread::mutex> guard(writeLock);
|
|
while(threadPool.size()>size_t(writeCount))//missed it somehow already.
|
|
{
|
|
writesDone.wait(writeLock); //if not, wait a bit
|
|
}
|
|
}
|
|
|
|
lightThreadDispatch::~lightThreadDispatch()
|
|
{
|
|
shutdown();
|
|
}
|