]> git.lizzy.rs Git - dragonfireclient.git/blobdiff - src/game.cpp
Make object hit delay smaller for fun; actually it should be roughly zero and hit...
[dragonfireclient.git] / src / game.cpp
index 4d9b0c3f6325d044787c8c3e82bf302af60b6d86..a7e1ccd8200cdbb71dcc113bf4bd070dc4f140f0 100644 (file)
@@ -43,13 +43,12 @@ with this program; if not, write to the Free Software Foundation, Inc.,
 #include "gettext.h"
 #include "log.h"
 #include "filesys.h"
-
-/*
-       TODO: Move content-aware stuff to separate file by adding properties
-             and virtual interfaces
-*/
-#include "content_mapnode.h"
-#include "content_nodemeta.h"
+// Needed for determining pointing to nodes
+#include "nodedef.h"
+#include "nodemetadata.h"
+#include "main.h" // For g_settings
+#include "tooldef.h"
+#include "tile.h" // For TextureSource
 
 /*
        Setting this to 1 enables a special camera mode that forces
@@ -114,9 +113,9 @@ struct TextDestChat : public TextDest
        Client *m_client;
 };
 
-struct TextDestSignNode : public TextDest
+struct TextDestNodeMetadata : public TextDest
 {
-       TextDestSignNode(v3s16 p, Client *client)
+       TextDestNodeMetadata(v3s16 p, Client *client)
        {
                m_p = p;
                m_client = client;
@@ -157,6 +156,7 @@ class MainRespawnInitiator: public IRespawnInitiator
        Hotbar draw routine
 */
 void draw_hotbar(video::IVideoDriver *driver, gui::IGUIFont *font,
+               ITextureSource *tsrc,
                v2s32 centerlowerpos, s32 imgsize, s32 itemcount,
                Inventory *inventory, s32 halfheartcount)
 {
@@ -248,16 +248,16 @@ void draw_hotbar(video::IVideoDriver *driver, gui::IGUIFont *font,
 
                if(item != NULL)
                {
-                       drawInventoryItem(driver, font, item, rect, NULL);
+                       drawInventoryItem(driver, font, item, rect, NULL, tsrc);
                }
        }
        
        /*
                Draw hearts
        */
+       video::ITexture *heart_texture = tsrc->getTextureRaw("heart.png");
+       if(heart_texture)
        {
-               video::ITexture *heart_texture =
-                               driver->getTexture(getTexturePath("heart.png").c_str());
                v2s32 p = pos + v2s32(0, -20);
                for(s32 i=0; i<halfheartcount/2; i++)
                {
@@ -320,7 +320,7 @@ void getPointedNode(Client *client, v3f player_position,
                try
                {
                        n = client->getNode(v3s16(x,y,z));
-                       if(content_pointable(n.getContent()) == false)
+                       if(client->getNodeDefManager()->get(n).pointable == false)
                                continue;
                }
                catch(InvalidPositionException &e)
@@ -342,57 +342,73 @@ void getPointedNode(Client *client, v3f player_position,
                        v3s16(-1,0,0), // left
                };
                
-               /*
-                       Meta-objects
-               */
-               if(n.getContent() == CONTENT_TORCH)
-               {
-                       v3s16 dir = unpackDir(n.param2);
-                       v3f dir_f = v3f(dir.X, dir.Y, dir.Z);
-                       dir_f *= BS/2 - BS/6 - BS/20;
-                       v3f cpf = npf + dir_f;
-                       f32 distance = (cpf - camera_position).getLength();
+               const ContentFeatures &f = client->getNodeDefManager()->get(n);
+               
+               if(f.selection_box.type == NODEBOX_FIXED)
+               {
+                       core::aabbox3d<f32> box = f.selection_box.fixed;
+                       box.MinEdge += npf;
+                       box.MaxEdge += npf;
+
+                       v3s16 facedirs[6] = {
+                               v3s16(-1,0,0),
+                               v3s16(1,0,0),
+                               v3s16(0,-1,0),
+                               v3s16(0,1,0),
+                               v3s16(0,0,-1),
+                               v3s16(0,0,1),
+                       };
 
-                       core::aabbox3d<f32> box;
-                       
-                       // bottom
-                       if(dir == v3s16(0,-1,0))
-                       {
-                               box = core::aabbox3d<f32>(
-                                       npf - v3f(BS/6, BS/2, BS/6),
-                                       npf + v3f(BS/6, -BS/2+BS/3*2, BS/6)
-                               );
-                       }
-                       // top
-                       else if(dir == v3s16(0,1,0))
-                       {
-                               box = core::aabbox3d<f32>(
-                                       npf - v3f(BS/6, -BS/2+BS/3*2, BS/6),
-                                       npf + v3f(BS/6, BS/2, BS/6)
-                               );
-                       }
-                       // side
-                       else
-                       {
-                               box = core::aabbox3d<f32>(
-                                       cpf - v3f(BS/6, BS/3, BS/6),
-                                       cpf + v3f(BS/6, BS/3, BS/6)
-                               );
-                       }
+                       core::aabbox3d<f32> faceboxes[6] = {
+                               // X-
+                               core::aabbox3d<f32>(
+                                       box.MinEdge.X, box.MinEdge.Y, box.MinEdge.Z,
+                                       box.MinEdge.X+d, box.MaxEdge.Y, box.MaxEdge.Z
+                               ),
+                               // X+
+                               core::aabbox3d<f32>(
+                                       box.MaxEdge.X-d, box.MinEdge.Y, box.MinEdge.Z,
+                                       box.MaxEdge.X, box.MaxEdge.Y, box.MaxEdge.Z
+                               ),
+                               // Y-
+                               core::aabbox3d<f32>(
+                                       box.MinEdge.X, box.MinEdge.Y, box.MinEdge.Z,
+                                       box.MaxEdge.X, box.MinEdge.Y+d, box.MaxEdge.Z
+                               ),
+                               // Y+
+                               core::aabbox3d<f32>(
+                                       box.MinEdge.X, box.MaxEdge.Y-d, box.MinEdge.Z,
+                                       box.MaxEdge.X, box.MaxEdge.Y, box.MaxEdge.Z
+                               ),
+                               // Z-
+                               core::aabbox3d<f32>(
+                                       box.MinEdge.X, box.MinEdge.Y, box.MinEdge.Z,
+                                       box.MaxEdge.X, box.MaxEdge.Y, box.MinEdge.Z+d
+                               ),
+                               // Z+
+                               core::aabbox3d<f32>(
+                                       box.MinEdge.X, box.MinEdge.Y, box.MaxEdge.Z-d,
+                                       box.MaxEdge.X, box.MaxEdge.Y, box.MaxEdge.Z
+                               ),
+                       };
 
-                       if(distance < mindistance)
+                       for(u16 i=0; i<6; i++)
                        {
-                               if(box.intersectsWithLine(shootline))
-                               {
-                                       nodefound = true;
-                                       nodepos = np;
-                                       neighbourpos = np;
-                                       mindistance = distance;
-                                       nodehilightbox = box;
-                               }
+                               v3f facedir_f(facedirs[i].X, facedirs[i].Y, facedirs[i].Z);
+                               v3f centerpoint = npf + facedir_f * BS/2;
+                               f32 distance = (centerpoint - camera_position).getLength();
+                               if(distance >= mindistance)
+                                       continue;
+                               if(!faceboxes[i].intersectsWithLine(shootline))
+                                       continue;
+                               nodefound = true;
+                               nodepos = np;
+                               neighbourpos = np+facedirs[i];
+                               mindistance = distance;
+                               nodehilightbox = box;
                        }
                }
-               else if(n.getContent() == CONTENT_SIGN_WALL)
+               else if(f.selection_box.type == NODEBOX_WALLMOUNTED)
                {
                        v3s16 dir = unpackDir(n.param2);
                        v3f dir_f = v3f(dir.X, dir.Y, dir.Z);
@@ -400,122 +416,43 @@ void getPointedNode(Client *client, v3f player_position,
                        v3f cpf = npf + dir_f;
                        f32 distance = (cpf - camera_position).getLength();
 
-                       v3f vertices[4] =
-                       {
-                               v3f(BS*0.42,-BS*0.35,-BS*0.4),
-                               v3f(BS*0.49, BS*0.35, BS*0.4),
-                       };
-
-                       for(s32 i=0; i<2; i++)
-                       {
-                               if(dir == v3s16(1,0,0))
-                                       vertices[i].rotateXZBy(0);
-                               if(dir == v3s16(-1,0,0))
-                                       vertices[i].rotateXZBy(180);
-                               if(dir == v3s16(0,0,1))
-                                       vertices[i].rotateXZBy(90);
-                               if(dir == v3s16(0,0,-1))
-                                       vertices[i].rotateXZBy(-90);
-                               if(dir == v3s16(0,-1,0))
-                                       vertices[i].rotateXYBy(-90);
-                               if(dir == v3s16(0,1,0))
-                                       vertices[i].rotateXYBy(90);
-
-                               vertices[i] += npf;
-                       }
-
                        core::aabbox3d<f32> box;
-
-                       box = core::aabbox3d<f32>(vertices[0]);
-                       box.addInternalPoint(vertices[1]);
-
-                       if(distance < mindistance)
-                       {
-                               if(box.intersectsWithLine(shootline))
-                               {
-                                       nodefound = true;
-                                       nodepos = np;
-                                       neighbourpos = np;
-                                       mindistance = distance;
-                                       nodehilightbox = box;
-                               }
+                       
+                       // top
+                       if(dir == v3s16(0,1,0)){
+                               box = f.selection_box.wall_top;
                        }
-               }
-
-               else if(n.getContent() == CONTENT_LADDER)
-               {
-                       v3s16 dir = unpackDir(n.param2);
-                       v3f dir_f = v3f(dir.X, dir.Y, dir.Z);
-                       dir_f *= BS/2 - BS/6 - BS/20;
-                       v3f cpf = npf + dir_f;
-                       f32 distance = (cpf - camera_position).getLength();
-
-                       v3f vertices[4] =
-                       {
-                               v3f(BS*0.42,-BS/2,-BS/2),
-                               v3f(BS*0.49, BS/2, BS/2),
-                       };
-
-                       for(s32 i=0; i<2; i++)
-                       {
-                               if(dir == v3s16(1,0,0))
-                                       vertices[i].rotateXZBy(0);
-                               if(dir == v3s16(-1,0,0))
-                                       vertices[i].rotateXZBy(180);
-                               if(dir == v3s16(0,0,1))
-                                       vertices[i].rotateXZBy(90);
-                               if(dir == v3s16(0,0,-1))
-                                       vertices[i].rotateXZBy(-90);
-                               if(dir == v3s16(0,-1,0))
-                                       vertices[i].rotateXYBy(-90);
-                               if(dir == v3s16(0,1,0))
-                                       vertices[i].rotateXYBy(90);
-
-                               vertices[i] += npf;
+                       // bottom
+                       else if(dir == v3s16(0,-1,0)){
+                               box = f.selection_box.wall_bottom;
                        }
+                       // side
+                       else{
+                               v3f vertices[2] =
+                               {
+                                       f.selection_box.wall_side.MinEdge,
+                                       f.selection_box.wall_side.MaxEdge
+                               };
 
-                       core::aabbox3d<f32> box;
-
-                       box = core::aabbox3d<f32>(vertices[0]);
-                       box.addInternalPoint(vertices[1]);
-
-                       if(distance < mindistance)
-                       {
-                               if(box.intersectsWithLine(shootline))
+                               for(s32 i=0; i<2; i++)
                                {
-                                       nodefound = true;
-                                       nodepos = np;
-                                       neighbourpos = np;
-                                       mindistance = distance;
-                                       nodehilightbox = box;
+                                       if(dir == v3s16(-1,0,0))
+                                               vertices[i].rotateXZBy(0);
+                                       if(dir == v3s16(1,0,0))
+                                               vertices[i].rotateXZBy(180);
+                                       if(dir == v3s16(0,0,-1))
+                                               vertices[i].rotateXZBy(90);
+                                       if(dir == v3s16(0,0,1))
+                                               vertices[i].rotateXZBy(-90);
                                }
-                       }
-               }
-               else if(n.getContent() == CONTENT_RAIL)
-               {
-                       v3s16 dir = unpackDir(n.param0);
-                       v3f dir_f = v3f(dir.X, dir.Y, dir.Z);
-                       dir_f *= BS/2 - BS/6 - BS/20;
-                       v3f cpf = npf + dir_f;
-                       f32 distance = (cpf - camera_position).getLength();
 
-                       float d = (float)BS/16;
-                       v3f vertices[4] =
-                       {
-                               v3f(BS/2, -BS/2+d, -BS/2),
-                               v3f(-BS/2, -BS/2, BS/2),
-                       };
-
-                       for(s32 i=0; i<2; i++)
-                       {
-                               vertices[i] += npf;
+                               box = core::aabbox3d<f32>(vertices[0]);
+                               box.addInternalPoint(vertices[1]);
                        }
 
-                       core::aabbox3d<f32> box;
-
-                       box = core::aabbox3d<f32>(vertices[0]);
-                       box.addInternalPoint(vertices[1]);
-
+                       box.MinEdge += npf;
+                       box.MaxEdge += npf;
+                       
                        if(distance < mindistance)
                        {
                                if(box.intersectsWithLine(shootline))
@@ -528,10 +465,7 @@ void getPointedNode(Client *client, v3f player_position,
                                }
                        }
                }
-               /*
-                       Regular blocks
-               */
-               else
+               else // NODEBOX_REGULAR
                {
                        for(u16 i=0; i<6; i++)
                        {
@@ -584,7 +518,7 @@ void getPointedNode(Client *client, v3f player_position,
        } // for coords
 }
 
-void update_skybox(video::IVideoDriver* driver,
+void update_skybox(video::IVideoDriver* driver, ITextureSource *tsrc,
                scene::ISceneManager* smgr, scene::ISceneNode* &skybox,
                float brightness)
 {
@@ -600,32 +534,32 @@ void update_skybox(video::IVideoDriver* driver,
        if(brightness >= 0.5)
        {
                skybox = smgr->addSkyBoxSceneNode(
-                       driver->getTexture(getTexturePath("skybox2.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox3.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1.png").c_str()));
+                       tsrc->getTextureRaw("skybox2.png"),
+                       tsrc->getTextureRaw("skybox3.png"),
+                       tsrc->getTextureRaw("skybox1.png"),
+                       tsrc->getTextureRaw("skybox1.png"),
+                       tsrc->getTextureRaw("skybox1.png"),
+                       tsrc->getTextureRaw("skybox1.png"));
        }
        else if(brightness >= 0.2)
        {
                skybox = smgr->addSkyBoxSceneNode(
-                       driver->getTexture(getTexturePath("skybox2_dawn.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox3_dawn.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_dawn.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_dawn.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_dawn.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_dawn.png").c_str()));
+                       tsrc->getTextureRaw("skybox2_dawn.png"),
+                       tsrc->getTextureRaw("skybox3_dawn.png"),
+                       tsrc->getTextureRaw("skybox1_dawn.png"),
+                       tsrc->getTextureRaw("skybox1_dawn.png"),
+                       tsrc->getTextureRaw("skybox1_dawn.png"),
+                       tsrc->getTextureRaw("skybox1_dawn.png"));
        }
        else
        {
                skybox = smgr->addSkyBoxSceneNode(
-                       driver->getTexture(getTexturePath("skybox2_night.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox3_night.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_night.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_night.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_night.png").c_str()),
-                       driver->getTexture(getTexturePath("skybox1_night.png").c_str()));
+                       tsrc->getTextureRaw("skybox2_night.png"),
+                       tsrc->getTextureRaw("skybox3_night.png"),
+                       tsrc->getTextureRaw("skybox1_night.png"),
+                       tsrc->getTextureRaw("skybox1_night.png"),
+                       tsrc->getTextureRaw("skybox1_night.png"),
+                       tsrc->getTextureRaw("skybox1_night.png"));
        }
 }
 
@@ -696,11 +630,18 @@ void the_game(
        /*
                Draw "Loading" screen
        */
-       /*gui::IGUIStaticText *gui_loadingtext = */
-       //draw_load_screen(L"Loading and connecting...", driver, font);
 
        draw_load_screen(L"Loading...", driver, font);
        
+       // Create texture source
+       IWritableTextureSource *tsrc = createTextureSource(device);
+       
+       // These will be filled by data received from the server
+       // Create tool definition manager
+       IWritableToolDefManager *tooldef = createToolDefManager();
+       // Create node definition manager
+       IWritableNodeDefManager *nodedef = createNodeDefManager();
+
        /*
                Create server.
                SharedPtr will delete it when it goes out of scope.
@@ -712,6 +653,8 @@ void the_game(
                server = new Server(map_dir, configpath);
                server->start(port);
        }
+
+       { // Client scope
        
        /*
                Create client
@@ -719,8 +662,14 @@ void the_game(
 
        draw_load_screen(L"Creating client...", driver, font);
        infostream<<"Creating client"<<std::endl;
+       
        MapDrawControl draw_control;
-       Client client(device, playername.c_str(), password, draw_control);
+
+       Client client(device, playername.c_str(), password, draw_control,
+                       tsrc, tooldef, nodedef);
+       
+       // Client acts as our GameDef
+       IGameDef *gamedef = &client;
                        
        draw_load_screen(L"Resolving address...", driver, font);
        Address connect_address(0,0,0,0, port);
@@ -748,54 +697,51 @@ void the_game(
        connect_address.print(&infostream);
        infostream<<std::endl;
        client.connect(connect_address);
-
-       bool could_connect = false;
        
+       /*
+               Wait for server to accept connection
+       */
+       bool could_connect = false;
        try{
+               float frametime = 0.033;
+               const float timeout = 10.0;
                float time_counter = 0.0;
                for(;;)
                {
-                       if(client.connectedAndInitialized())
-                       {
+                       // Update client and server
+                       client.step(frametime);
+                       if(server != NULL)
+                               server->step(frametime);
+                       
+                       // End condition
+                       if(client.connectedAndInitialized()){
                                could_connect = true;
                                break;
                        }
+                       // Break conditions
                        if(client.accessDenied())
-                       {
                                break;
-                       }
-                       // Wait for 10 seconds
-                       if(time_counter >= 10.0)
-                       {
+                       if(time_counter >= timeout)
                                break;
-                       }
                        
+                       // Display status
                        std::wostringstream ss;
                        ss<<L"Connecting to server... (timeout in ";
-                       ss<<(int)(10.0 - time_counter + 1.0);
+                       ss<<(int)(timeout - time_counter + 1.0);
                        ss<<L" seconds)";
                        draw_load_screen(ss.str(), driver, font);
-
-                       /*// Update screen
-                       driver->beginScene(true, true, video::SColor(255,0,0,0));
-                       guienv->drawAll();
-                       driver->endScene();*/
-
-                       // Update client and server
-
-                       client.step(0.1);
-
-                       if(server != NULL)
-                               server->step(0.1);
                        
                        // Delay a bit
-                       sleep_ms(100);
-                       time_counter += 0.1;
+                       sleep_ms(1000*frametime);
+                       time_counter += frametime;
                }
        }
        catch(con::PeerNotFoundException &e)
        {}
-
+       
+       /*
+               Handle failure to connect
+       */
        if(could_connect == false)
        {
                if(client.accessDenied())
@@ -812,13 +758,63 @@ void the_game(
                //gui_loadingtext->remove();
                return;
        }
+       
+       /*
+               Wait until content has been received
+       */
+       bool got_content = false;
+       {
+               float frametime = 0.033;
+               const float timeout = 5.0;
+               float time_counter = 0.0;
+               for(;;)
+               {
+                       // Update client and server
+                       client.step(frametime);
+                       if(server != NULL)
+                               server->step(frametime);
+                       
+                       // End condition
+                       if(client.texturesReceived() &&
+                                       client.tooldefReceived() &&
+                                       client.nodedefReceived()){
+                               got_content = true;
+                               break;
+                       }
+                       // Break conditions
+                       if(!client.connectedAndInitialized())
+                               break;
+                       if(time_counter >= timeout)
+                               break;
+                       
+                       // Display status
+                       std::wostringstream ss;
+                       ss<<L"Waiting content... (continuing anyway in ";
+                       ss<<(int)(timeout - time_counter + 1.0);
+                       ss<<L" seconds)\n";
+
+                       ss<<(client.tooldefReceived()?L"[X]":L"[  ]");
+                       ss<<L" Tool definitions\n";
+                       ss<<(client.nodedefReceived()?L"[X]":L"[  ]");
+                       ss<<L" Node definitions\n";
+                       //ss<<(client.texturesReceived()?L"[X]":L"[  ]");
+                       ss<<L"["<<(int)(client.textureReceiveProgress()*100+0.5)<<L"%] ";
+                       ss<<L" Textures\n";
+
+                       draw_load_screen(ss.str(), driver, font);
+                       
+                       // Delay a bit
+                       sleep_ms(1000*frametime);
+                       time_counter += frametime;
+               }
+       }
 
        /*
                Create skybox
        */
        float old_brightness = 1.0;
        scene::ISceneNode* skybox = NULL;
-       update_skybox(driver, smgr, skybox, 1.0);
+       update_skybox(driver, tsrc, smgr, skybox, 1.0);
        
        /*
                Create the camera node
@@ -872,7 +868,6 @@ void the_game(
                        L"",
                        core::rect<s32>(5, 5+(text_height+5)*1, 795, (5+text_height)*2),
                        false, false);
-       
        // At the middle of the screen
        // Object infos are shown in this
        gui::IGUIStaticText *guitext_info = guienv->addStaticText(
@@ -889,6 +884,15 @@ void the_game(
        //guitext_chat->setBackgroundColor(video::SColor(96,0,0,0));
        core::list<ChatLine> chat_lines;
        
+       // Profiler text (size is updated when text is updated)
+       gui::IGUIStaticText *guitext_profiler = guienv->addStaticText(
+                       L"<Profiler>",
+                       core::rect<s32>(6, 4+(text_height+5)*2, 400,
+                       (text_height+5)*2 + text_height*35),
+                       false, false);
+       guitext_profiler->setBackgroundColor(video::SColor(80,0,0,0));
+       guitext_profiler->setVisible(false);
+       
        /*GUIQuickInventory *quick_inventory = new GUIQuickInventory
                        (guienv, NULL, v2s32(10, 70), 5, &local_inventory);*/
        /*GUIQuickInventory *quick_inventory = new GUIQuickInventory
@@ -925,23 +929,30 @@ void the_game(
        // A test
        //throw con::PeerNotFoundException("lol");
 
+       float brightness = 1.0;
+
        core::list<float> frametime_log;
 
        float nodig_delay_counter = 0.0;
        float dig_time = 0.0;
        u16 dig_index = 0;
        v3s16 nodepos_old(-32768,-32768,-32768);
+       bool ldown_for_dig = false;
 
        float damage_flash_timer = 0;
        s16 farmesh_range = 20*MAP_BLOCKSIZE;
 
-       const float object_hit_delay = 0.5;
+       const float object_hit_delay = 0.2;
        float object_hit_delay_timer = 0.0;
        
        bool invert_mouse = g_settings->getBool("invert_mouse");
 
        bool respawn_menu_active = false;
 
+       bool show_profiler = false;
+       bool force_fog_off = false;
+       bool disable_camera_update = false;
+
        /*
                Main loop
        */
@@ -985,7 +996,7 @@ void the_game(
                /*
                        Process TextureSource's queue
                */
-               ((TextureSource*)g_texturesource)->processQueue();
+               tsrc->processQueue();
 
                /*
                        Random calculations
@@ -1162,14 +1173,32 @@ void the_game(
                */
                float profiler_print_interval =
                                g_settings->getFloat("profiler_print_interval");
-               if(profiler_print_interval != 0)
+               bool print_to_log = true;
+               if(profiler_print_interval == 0){
+                       print_to_log = false;
+                       profiler_print_interval = 5;
+               }
+               if(m_profiler_interval.step(dtime, profiler_print_interval))
                {
-                       if(m_profiler_interval.step(0.030, profiler_print_interval))
-                       {
+                       if(print_to_log){
                                infostream<<"Profiler:"<<std::endl;
                                g_profiler->print(infostream);
-                               g_profiler->clear();
                        }
+
+                       std::ostringstream os(std::ios_base::binary);
+                       g_profiler->print(os);
+                       std::wstring text = narrow_to_wide(os.str());
+                       guitext_profiler->setText(text.c_str());
+
+                       g_profiler->clear();
+                       
+                       s32 w = font->getDimension(text.c_str()).Width;
+                       if(w < 400)
+                               w = 400;
+                       core::rect<s32> rect(6, 4+(text_height+5)*2, 12+w,
+                                       8+(text_height+5)*2 +
+                                       font->getDimension(text.c_str()).Height);
+                       guitext_profiler->setRelativePosition(rect);
                }
 
                /*
@@ -1197,7 +1226,7 @@ void the_game(
                                new GUIInventoryMenu(guienv, guiroot, -1,
                                        &g_menumgr, v2s16(8,7),
                                        client.getInventoryContext(),
-                                       &client);
+                                       &client, tsrc);
 
                        core::array<GUIInventoryMenu::DrawSpec> draw_spec;
                        draw_spec.push_back(GUIInventoryMenu::DrawSpec(
@@ -1299,6 +1328,31 @@ void the_game(
                                image->drop(); 
                        }                        
                }
+               else if(input->wasKeyDown(getKeySetting("keymap_toggle_profiler")))
+               {
+                       show_profiler = !show_profiler;
+                       guitext_profiler->setVisible(show_profiler);
+                       if(show_profiler)
+                               chat_lines.push_back(ChatLine(L"Profiler disabled"));
+                       else
+                               chat_lines.push_back(ChatLine(L"Profiler enabled"));
+               }
+               else if(input->wasKeyDown(getKeySetting("keymap_toggle_force_fog_off")))
+               {
+                       force_fog_off = !force_fog_off;
+                       if(force_fog_off)
+                               chat_lines.push_back(ChatLine(L"Fog disabled"));
+                       else
+                               chat_lines.push_back(ChatLine(L"Fog enabled"));
+               }
+               else if(input->wasKeyDown(getKeySetting("keymap_toggle_update_camera")))
+               {
+                       disable_camera_update = !disable_camera_update;
+                       if(disable_camera_update)
+                               chat_lines.push_back(ChatLine(L"Camera update disabled"));
+                       else
+                               chat_lines.push_back(ChatLine(L"Camera update enabled"));
+               }
 
                // Item selection with mouse wheel
                {
@@ -1529,6 +1583,10 @@ void the_game(
                                        player->setPosition(player->getPosition() + v3f(0,-BS,0));
                                        camera.update(player, busytime, screensize);*/
                                }
+                               else if(event.type == CE_TEXTURES_UPDATED)
+                               {
+                                       update_skybox(driver, tsrc, smgr, skybox, brightness);
+                               }
                        }
                }
                
@@ -1543,11 +1601,10 @@ void the_game(
                v3f camera_direction = camera.getDirection();
                f32 camera_fov = camera.getFovMax();
                
-               if(FIELD_OF_VIEW_TEST)
-                       client.updateCamera(v3f(0,0,0), v3f(0,0,1), camera_fov);
-               else
+               if(!disable_camera_update){
                        client.updateCamera(camera_position,
                                camera_direction, camera_fov);
+               }
 
                //timer2.stop();
                //TimeTaker //timer3("//timer3");
@@ -1570,7 +1627,7 @@ void the_game(
                bool left_punch = false;
                bool left_punch_muted = false;
 
-               if(selected_active_object != NULL)
+               if(selected_active_object != NULL && !ldown_for_dig)
                {
                        /* Clear possible cracking animation */
                        if(nodepos_old != v3s16(-32768,-32768,-32768))
@@ -1654,6 +1711,7 @@ void the_game(
                                client.clearTempMod(nodepos_old);
                                dig_time = 0.0;
                                nodepos_old = v3s16(-32768,-32768,-32768);
+                               ldown_for_dig = false;
                        }
                } else {
                        /*
@@ -1682,6 +1740,7 @@ void the_game(
                        {
                                client.clearTempMod(nodepos);
                                dig_time = 0.0;
+                               ldown_for_dig = false;
                        }
                        
                        if(nodig_delay_counter > 0.0)
@@ -1712,6 +1771,7 @@ void the_game(
                                if(input->getLeftClicked())
                                {
                                        client.setTempMod(nodepos, NodeMod(NODEMOD_CRACK, 0));
+                                       ldown_for_dig = true;
                                }
                                if(input->getLeftState())
                                {
@@ -1732,8 +1792,10 @@ void the_game(
 
                                        // Get digging properties for material and tool
                                        content_t material = n.getContent();
+                                       ToolDiggingProperties tp =
+                                                       tooldef->getDiggingProperties(toolname);
                                        DiggingProperties prop =
-                                                       getDiggingProperties(material, toolname);
+                                                       getDiggingProperties(material, &tp, nodedef);
                                        
                                        float dig_time_complete = 0.0;
 
@@ -1835,27 +1897,26 @@ void the_game(
                                                new GUIInventoryMenu(guienv, guiroot, -1,
                                                        &g_menumgr, invsize,
                                                        client.getInventoryContext(),
-                                                       &client);
+                                                       &client, tsrc);
                                        menu->setDrawSpec(draw_spec);
                                        menu->drop();
                                }
-                               else if(meta && meta->typeId() == CONTENT_SIGN_WALL && !random_input)
+                               // If metadata provides text input, activate text input
+                               else if(meta && meta->allowsTextInput() && !random_input)
                                {
-                                       infostream<<"Sign node right-clicked"<<std::endl;
-                                       
-                                       SignNodeMetadata *signmeta = (SignNodeMetadata*)meta;
+                                       infostream<<"Launching metadata text input"<<std::endl;
                                        
                                        // Get a new text for it
 
-                                       TextDest *dest = new TextDestSignNode(nodepos, &client);
+                                       TextDest *dest = new TextDestNodeMetadata(nodepos, &client);
 
-                                       std::wstring wtext =
-                                                       narrow_to_wide(signmeta->getText());
+                                       std::wstring wtext = narrow_to_wide(meta->getText());
 
                                        (new GUITextInputMenu(guienv, guiroot, -1,
                                                        &g_menumgr, dest,
                                                        wtext))->drop();
                                }
+                               // Otherwise report right click to server
                                else
                                {
                                        client.groundAction(1, nodepos, neighbourpos, g_selected_item);
@@ -1881,6 +1942,7 @@ void the_game(
                        infostream<<"Left button released (stopped digging)"
                                        <<std::endl;
                        client.groundAction(2, v3s16(0,0,0), v3s16(0,0,0), 0);
+                       ldown_for_dig = false;
                }
                if(input->getRightReleased())
                {
@@ -1894,25 +1956,27 @@ void the_game(
                /*
                        Calculate stuff for drawing
                */
-
+               
+               /*
+                       Calculate general brightness
+               */
                u32 daynight_ratio = client.getDayNightRatio();
-               u8 l = decode_light((daynight_ratio * LIGHT_SUN) / 1000);
+               u8 light8 = decode_light((daynight_ratio * LIGHT_SUN) / 1000);
+               brightness = (float)light8/255.0;
                video::SColor bgcolor = video::SColor(
                                255,
-                               bgcolor_bright.getRed() * l / 255,
-                               bgcolor_bright.getGreen() * l / 255,
-                               bgcolor_bright.getBlue() * l / 255);
-                               /*skycolor.getRed() * l / 255,
-                               skycolor.getGreen() * l / 255,
-                               skycolor.getBlue() * l / 255);*/
-
-               float brightness = (float)l/255.0;
+                               bgcolor_bright.getRed() * brightness,
+                               bgcolor_bright.getGreen() * brightness,
+                               bgcolor_bright.getBlue() * brightness);
+                               /*skycolor.getRed() * brightness,
+                               skycolor.getGreen() * brightness,
+                               skycolor.getBlue() * brightness);*/
 
                /*
                        Update skybox
                */
                if(fabs(brightness - old_brightness) > 0.01)
-                       update_skybox(driver, smgr, skybox, brightness);
+                       update_skybox(driver, tsrc, smgr, skybox, brightness);
 
                /*
                        Update clouds
@@ -1947,7 +2011,7 @@ void the_game(
                        Fog
                */
                
-               if(g_settings->getBool("enable_fog") == true)
+               if(g_settings->getBool("enable_fog") == true && !force_fog_off)
                {
                        f32 range;
                        if(farmesh)
@@ -1957,10 +2021,11 @@ void the_game(
                        else
                        {
                                range = draw_control.wanted_range*BS + MAP_BLOCKSIZE*BS*1.5;
+                               range *= 0.9;
                                if(draw_control.range_all)
                                        range = 100000*BS;
-                               if(range < 50*BS)
-                                       range = range * 0.5 + 25*BS;
+                               /*if(range < 50*BS)
+                                       range = range * 0.5 + 25*BS;*/
                        }
 
                        driver->setFog(
@@ -2114,7 +2179,8 @@ void the_game(
 
                        guitext_chat->setRelativePosition(rect);
 
-                       if(chat_lines.size() == 0)
+                       // Don't show chat if empty or profiler is enabled
+                       if(chat_lines.size() == 0 || show_profiler)
                                guitext_chat->setVisible(false);
                        else
                                guitext_chat->setVisible(true);
@@ -2138,7 +2204,7 @@ void the_game(
                        InventoryItem *item = NULL;
                        if(mlist != NULL)
                                item = mlist->getItem(g_selected_item);
-                       camera.wield(item);
+                       camera.wield(item, gamedef);
                }
                
                /*
@@ -2261,7 +2327,8 @@ void the_game(
                        Draw hotbar
                */
                {
-                       draw_hotbar(driver, font, v2s32(displaycenter.X, screensize.Y),
+                       draw_hotbar(driver, font, tsrc,
+                                       v2s32(displaycenter.X, screensize.Y),
                                        hotbar_imagesize, hotbar_itemcount, &local_inventory,
                                        client.getHP());
                }
@@ -2328,6 +2395,12 @@ void the_game(
                driver->endScene();
                gui_shuttingdowntext->remove();*/
        }
+
+       } // Client scope (must be destructed before destructing *def and tsrc
+
+       delete tooldef;
+       delete tsrc;
+       delete nodedef;
 }