2010-04-02 19:52:46 -06:00
|
|
|
/*
|
2011-06-16 15:53:39 -06:00
|
|
|
https://github.com/peterix/dfhack
|
|
|
|
Copyright (c) 2009-2011 Petr Mrázek (peterix@gmail.com)
|
2010-04-02 19:52:46 -06:00
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any
|
|
|
|
damages arising from the use of this software.
|
|
|
|
|
|
|
|
Permission is granted to anyone to use this software for any
|
|
|
|
purpose, including commercial applications, and to alter it and
|
|
|
|
redistribute it freely, subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must
|
|
|
|
not claim that you wrote the original software. If you use this
|
|
|
|
software in a product, an acknowledgment in the product documentation
|
|
|
|
would be appreciated but is not required.
|
|
|
|
|
|
|
|
2. Altered source versions must be plainly marked as such, and
|
|
|
|
must not be misrepresented as being the original software.
|
|
|
|
|
|
|
|
3. This notice may not be removed or altered from any source
|
|
|
|
distribution.
|
|
|
|
*/
|
|
|
|
|
2011-06-16 15:53:39 -06:00
|
|
|
|
2010-05-26 04:24:45 -06:00
|
|
|
#include "Internal.h"
|
2011-04-10 02:19:15 -06:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
#include <map>
|
|
|
|
using namespace std;
|
|
|
|
|
2011-12-31 04:48:42 -07:00
|
|
|
#include "modules/Gui.h"
|
|
|
|
#include "MemAccess.h"
|
|
|
|
#include "VersionInfo.h"
|
|
|
|
#include "Types.h"
|
|
|
|
#include "Error.h"
|
2011-03-18 01:53:59 -06:00
|
|
|
#include "ModuleFactory.h"
|
2011-12-31 04:48:42 -07:00
|
|
|
#include "Core.h"
|
2011-12-31 05:09:12 -07:00
|
|
|
#include "PluginManager.h"
|
2010-04-02 19:52:46 -06:00
|
|
|
using namespace DFHack;
|
|
|
|
|
2011-12-31 05:09:12 -07:00
|
|
|
#include "DataDefs.h"
|
2012-01-07 08:21:07 -07:00
|
|
|
#include "df/world.h"
|
2011-12-31 05:09:12 -07:00
|
|
|
#include "df/cursor.h"
|
|
|
|
#include "df/viewscreen_dwarfmodest.h"
|
2012-01-07 08:21:07 -07:00
|
|
|
#include "df/ui.h"
|
|
|
|
#include "df/job.h"
|
|
|
|
#include "df/ui_build_selector.h"
|
|
|
|
#include "df/building_workshopst.h"
|
|
|
|
#include "df/building_furnacest.h"
|
|
|
|
|
|
|
|
using namespace df::enums;
|
2011-12-31 02:25:46 -07:00
|
|
|
|
|
|
|
// Predefined common guard functions
|
|
|
|
|
|
|
|
bool DFHack::default_hotkey(Core *, df::viewscreen *top)
|
|
|
|
{
|
|
|
|
// Default hotkey guard function
|
|
|
|
for (;top ;top = top->parent)
|
|
|
|
if (strict_virtual_cast<df::viewscreen_dwarfmodest>(top))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DFHack::dwarfmode_hotkey(Core *, df::viewscreen *top)
|
|
|
|
{
|
|
|
|
// Require the main dwarf mode screen
|
|
|
|
return !!strict_virtual_cast<df::viewscreen_dwarfmodest>(top);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DFHack::cursor_hotkey(Core *c, df::viewscreen *top)
|
|
|
|
{
|
|
|
|
if (!dwarfmode_hotkey(c, top))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Also require the cursor.
|
|
|
|
if (!df::global::cursor || df::global::cursor->x == -30000)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-01-07 08:21:07 -07:00
|
|
|
bool DFHack::workshop_job_hotkey(Core *c, df::viewscreen *top)
|
|
|
|
{
|
|
|
|
using namespace ui_sidebar_mode;
|
|
|
|
using df::global::ui;
|
|
|
|
using df::global::world;
|
|
|
|
using df::global::ui_workshop_in_add;
|
|
|
|
using df::global::ui_workshop_job_cursor;
|
|
|
|
|
|
|
|
if (!dwarfmode_hotkey(c,top))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (ui->main.mode) {
|
|
|
|
case QueryBuilding:
|
|
|
|
{
|
|
|
|
if (!ui_workshop_job_cursor) // allow missing
|
|
|
|
return false;
|
|
|
|
|
|
|
|
df::building *selected = world->selected_building;
|
|
|
|
if (!virtual_cast<df::building_workshopst>(selected) &&
|
|
|
|
!virtual_cast<df::building_furnacest>(selected))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// No jobs?
|
|
|
|
if (selected->jobs.empty() ||
|
|
|
|
selected->jobs[0]->job_type == job_type::DestroyBuilding)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Add job gui activated?
|
|
|
|
if (ui_workshop_in_add && *ui_workshop_in_add)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DFHack::build_selector_hotkey(Core *c, df::viewscreen *top)
|
|
|
|
{
|
|
|
|
using namespace ui_sidebar_mode;
|
|
|
|
using df::global::ui;
|
|
|
|
using df::global::ui_build_selector;
|
|
|
|
|
|
|
|
if (!dwarfmode_hotkey(c,top))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
switch (ui->main.mode) {
|
|
|
|
case Build:
|
|
|
|
{
|
|
|
|
if (!ui_build_selector) // allow missing
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Not selecting, or no choices?
|
|
|
|
if (ui_build_selector->building_type < 0 ||
|
|
|
|
ui_build_selector->stage != 2 ||
|
|
|
|
ui_build_selector->choices.empty())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
df::job *DFHack::getSelectedWorkshopJob(Core *c, bool quiet)
|
|
|
|
{
|
|
|
|
using df::global::world;
|
|
|
|
using df::global::ui_workshop_job_cursor;
|
|
|
|
|
|
|
|
if (!workshop_job_hotkey(c, c->getTopViewscreen())) {
|
|
|
|
if (!quiet)
|
|
|
|
c->con.printerr("Not in a workshop, or no job is highlighted.\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
df::building *selected = world->selected_building;
|
|
|
|
int idx = *ui_workshop_job_cursor;
|
|
|
|
|
|
|
|
if (idx < 0 || idx >= selected->jobs.size())
|
|
|
|
{
|
|
|
|
c->con.printerr("Invalid job cursor index: %d\n", idx);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return selected->jobs[idx];
|
|
|
|
}
|
|
|
|
|
2011-12-31 02:25:46 -07:00
|
|
|
//
|
|
|
|
|
2011-06-17 07:02:43 -06:00
|
|
|
Module* DFHack::createGui()
|
2011-03-18 01:53:59 -06:00
|
|
|
{
|
2011-06-17 07:02:43 -06:00
|
|
|
return new Gui();
|
2011-03-18 01:53:59 -06:00
|
|
|
}
|
|
|
|
|
2010-04-04 16:48:19 -06:00
|
|
|
struct Gui::Private
|
|
|
|
{
|
2010-08-31 04:59:48 -06:00
|
|
|
Private()
|
|
|
|
{
|
2011-07-09 03:33:58 -06:00
|
|
|
Started = false;
|
2011-06-14 22:09:24 -06:00
|
|
|
StartedScreen = false;
|
2011-09-25 19:39:27 -06:00
|
|
|
mouse_xy_offset = 0;
|
|
|
|
designation_xyz_offset = 0;
|
2010-08-31 04:59:48 -06:00
|
|
|
}
|
2011-03-18 04:09:26 -06:00
|
|
|
|
|
|
|
bool Started;
|
2012-01-03 17:45:11 -07:00
|
|
|
int32_t * window_x_offset;
|
|
|
|
int32_t * window_y_offset;
|
|
|
|
int32_t * window_z_offset;
|
|
|
|
struct xyz
|
|
|
|
{
|
|
|
|
int32_t x;
|
|
|
|
int32_t y;
|
|
|
|
int32_t z;
|
|
|
|
} * cursor_xyz_offset, * designation_xyz_offset;
|
|
|
|
struct xy
|
|
|
|
{
|
|
|
|
int32_t x;
|
|
|
|
int32_t y;
|
|
|
|
} * mouse_xy_offset, * window_dims_offset;
|
2011-03-18 04:09:26 -06:00
|
|
|
|
|
|
|
bool StartedScreen;
|
2012-01-03 17:45:11 -07:00
|
|
|
void * screen_tiles_ptr_offset;
|
2011-03-18 04:09:26 -06:00
|
|
|
|
2010-04-18 13:30:02 -06:00
|
|
|
Process * owner;
|
2010-04-04 16:48:19 -06:00
|
|
|
};
|
|
|
|
|
2011-06-17 07:02:43 -06:00
|
|
|
Gui::Gui()
|
2010-04-04 16:48:19 -06:00
|
|
|
{
|
2011-06-17 07:02:43 -06:00
|
|
|
Core & c = Core::getInstance();
|
2010-04-04 16:48:19 -06:00
|
|
|
d = new Private;
|
2011-06-17 07:02:43 -06:00
|
|
|
d->owner = c.p;
|
|
|
|
VersionInfo * mem = c.vinfo;
|
2011-03-18 04:09:26 -06:00
|
|
|
OffsetGroup * OG_Gui = mem->getGroup("GUI");
|
2011-07-09 03:33:58 -06:00
|
|
|
|
|
|
|
// Setting up hotkeys
|
2010-08-31 04:59:48 -06:00
|
|
|
try
|
2011-06-14 22:09:24 -06:00
|
|
|
{
|
|
|
|
hotkeys = (hotkey_array *) OG_Gui->getAddress("hotkeys");
|
|
|
|
}
|
|
|
|
catch(Error::All &)
|
|
|
|
{
|
|
|
|
hotkeys = 0;
|
|
|
|
};
|
2011-07-09 03:33:58 -06:00
|
|
|
|
2011-07-31 19:31:52 -06:00
|
|
|
// Setting up init
|
|
|
|
try
|
|
|
|
{
|
|
|
|
init = (t_init *) OG_Gui->getAddress("init");
|
|
|
|
}
|
|
|
|
catch(Error::All &)
|
|
|
|
{
|
|
|
|
init = 0;
|
|
|
|
};
|
|
|
|
|
2011-07-09 03:33:58 -06:00
|
|
|
// Setting up menu state
|
2011-06-14 22:09:24 -06:00
|
|
|
try
|
2010-08-31 04:59:48 -06:00
|
|
|
{
|
2011-07-16 10:10:08 -06:00
|
|
|
df_menu_state = (uint32_t *) OG_Gui->getAddress("current_menu_state");
|
2010-08-31 04:59:48 -06:00
|
|
|
}
|
2011-07-09 03:33:58 -06:00
|
|
|
catch(Error::All &)
|
|
|
|
{
|
2011-07-16 10:10:08 -06:00
|
|
|
df_menu_state = 0;
|
2011-07-09 03:33:58 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
// Setting up the view screen stuff
|
2010-08-31 04:59:48 -06:00
|
|
|
try
|
|
|
|
{
|
2011-07-16 10:10:08 -06:00
|
|
|
df_interface = (t_interface *) OG_Gui->getAddress ("interface");
|
2010-08-31 04:59:48 -06:00
|
|
|
}
|
2011-07-09 03:33:58 -06:00
|
|
|
catch(exception &)
|
|
|
|
{
|
2011-07-16 10:10:08 -06:00
|
|
|
df_interface = 0;
|
2011-07-09 03:33:58 -06:00
|
|
|
};
|
|
|
|
|
2011-03-18 04:09:26 -06:00
|
|
|
OffsetGroup * OG_Position;
|
2010-08-31 04:59:48 -06:00
|
|
|
try
|
|
|
|
{
|
2011-03-18 04:09:26 -06:00
|
|
|
OG_Position = mem->getGroup("Position");
|
2012-01-03 17:45:11 -07:00
|
|
|
d->window_x_offset = (int32_t *) OG_Position->getAddress ("window_x");
|
|
|
|
d->window_y_offset = (int32_t *) OG_Position->getAddress ("window_y");
|
|
|
|
d->window_z_offset = (int32_t *) OG_Position->getAddress ("window_z");
|
|
|
|
d->cursor_xyz_offset = (Private::xyz *) OG_Position->getAddress ("cursor_xyz");
|
|
|
|
d->window_dims_offset = (Private::xy *) OG_Position->getAddress ("window_dims");
|
2011-03-18 04:09:26 -06:00
|
|
|
d->Started = true;
|
2010-08-31 04:59:48 -06:00
|
|
|
}
|
2011-03-18 04:09:26 -06:00
|
|
|
catch(Error::All &){};
|
|
|
|
try
|
|
|
|
{
|
2012-01-03 17:45:11 -07:00
|
|
|
d->mouse_xy_offset = (Private::xy *) OG_Position->getAddress ("mouse_xy");
|
|
|
|
}
|
|
|
|
catch(Error::All &)
|
|
|
|
{
|
|
|
|
d->mouse_xy_offset = 0;
|
|
|
|
};
|
|
|
|
try
|
|
|
|
{
|
|
|
|
d->designation_xyz_offset = (Private::xyz *) OG_Position->getAddress ("designation_xyz");
|
|
|
|
}
|
|
|
|
catch(Error::All &)
|
|
|
|
{
|
|
|
|
d->designation_xyz_offset = 0;
|
|
|
|
};
|
|
|
|
try
|
|
|
|
{
|
|
|
|
d->screen_tiles_ptr_offset = (void *) OG_Position->getAddress ("screen_tiles_pointer");
|
2011-03-18 04:09:26 -06:00
|
|
|
d->StartedScreen = true;
|
|
|
|
}
|
|
|
|
catch(Error::All &){};
|
2010-04-04 16:48:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
Gui::~Gui()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Gui::Start()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Gui::Finish()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-07-09 03:33:58 -06:00
|
|
|
t_viewscreen * Gui::GetCurrentScreen()
|
2010-04-02 19:52:46 -06:00
|
|
|
{
|
2011-07-16 10:10:08 -06:00
|
|
|
if(!df_interface)
|
2011-07-09 03:33:58 -06:00
|
|
|
return 0;
|
2011-07-16 10:10:08 -06:00
|
|
|
t_viewscreen * ws = &df_interface->view;
|
2011-07-09 03:33:58 -06:00
|
|
|
while(ws)
|
2010-04-02 19:52:46 -06:00
|
|
|
{
|
2011-07-09 03:33:58 -06:00
|
|
|
if(ws->child)
|
|
|
|
ws = ws->child;
|
|
|
|
else
|
|
|
|
return ws;
|
2010-04-02 19:52:46 -06:00
|
|
|
}
|
2011-07-09 03:33:58 -06:00
|
|
|
return 0;
|
2010-04-02 19:52:46 -06:00
|
|
|
}
|
2011-03-18 04:09:26 -06:00
|
|
|
|
|
|
|
bool Gui::getViewCoords (int32_t &x, int32_t &y, int32_t &z)
|
|
|
|
{
|
|
|
|
if (!d->Started) return false;
|
|
|
|
Process * p = d->owner;
|
|
|
|
|
|
|
|
p->readDWord (d->window_x_offset, (uint32_t &) x);
|
|
|
|
p->readDWord (d->window_y_offset, (uint32_t &) y);
|
|
|
|
p->readDWord (d->window_z_offset, (uint32_t &) z);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//FIXME: confine writing of coords to map bounds?
|
|
|
|
bool Gui::setViewCoords (const int32_t x, const int32_t y, const int32_t z)
|
|
|
|
{
|
|
|
|
if (!d->Started)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Process * p = d->owner;
|
|
|
|
|
|
|
|
p->writeDWord (d->window_x_offset, (uint32_t) x);
|
|
|
|
p->writeDWord (d->window_y_offset, (uint32_t) y);
|
|
|
|
p->writeDWord (d->window_z_offset, (uint32_t) z);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Gui::getCursorCoords (int32_t &x, int32_t &y, int32_t &z)
|
|
|
|
{
|
|
|
|
if(!d->Started) return false;
|
|
|
|
int32_t coords[3];
|
|
|
|
d->owner->read (d->cursor_xyz_offset, 3*sizeof (int32_t), (uint8_t *) coords);
|
|
|
|
x = coords[0];
|
|
|
|
y = coords[1];
|
|
|
|
z = coords[2];
|
|
|
|
if (x == -30000) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//FIXME: confine writing of coords to map bounds?
|
|
|
|
bool Gui::setCursorCoords (const int32_t x, const int32_t y, const int32_t z)
|
|
|
|
{
|
|
|
|
if (!d->Started) return false;
|
|
|
|
int32_t coords[3] = {x, y, z};
|
|
|
|
d->owner->write (d->cursor_xyz_offset, 3*sizeof (int32_t), (uint8_t *) coords);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:39:27 -06:00
|
|
|
bool Gui::getDesignationCoords (int32_t &x, int32_t &y, int32_t &z)
|
|
|
|
{
|
|
|
|
if(!d->designation_xyz_offset) return false;
|
|
|
|
int32_t coords[3];
|
|
|
|
d->owner->read (d->designation_xyz_offset, 3*sizeof (int32_t), (uint8_t *) coords);
|
|
|
|
x = coords[0];
|
|
|
|
y = coords[1];
|
|
|
|
z = coords[2];
|
|
|
|
if (x == -30000) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Gui::setDesignationCoords (const int32_t x, const int32_t y, const int32_t z)
|
|
|
|
{
|
|
|
|
if(!d->designation_xyz_offset) return false;
|
|
|
|
int32_t coords[3] = {x, y, z};
|
|
|
|
d->owner->write (d->designation_xyz_offset, 3*sizeof (int32_t), (uint8_t *) coords);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Gui::getMousePos (int32_t & x, int32_t & y)
|
|
|
|
{
|
|
|
|
if(!d->mouse_xy_offset) return false;
|
|
|
|
int32_t coords[2];
|
|
|
|
d->owner->read (d->mouse_xy_offset, 2*sizeof (int32_t), (uint8_t *) coords);
|
|
|
|
x = coords[0];
|
|
|
|
y = coords[1];
|
|
|
|
if(x == -1) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-03-18 04:09:26 -06:00
|
|
|
bool Gui::getWindowSize (int32_t &width, int32_t &height)
|
|
|
|
{
|
|
|
|
if(!d->Started) return false;
|
|
|
|
|
|
|
|
int32_t coords[2];
|
|
|
|
d->owner->read (d->window_dims_offset, 2*sizeof (int32_t), (uint8_t *) coords);
|
|
|
|
width = coords[0];
|
|
|
|
height = coords[1];
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Gui::getScreenTiles (int32_t width, int32_t height, t_screen screen[])
|
|
|
|
{
|
|
|
|
if(!d->StartedScreen) return false;
|
|
|
|
|
2012-01-03 17:45:11 -07:00
|
|
|
void * screen_addr = (void *) d->owner->readDWord(d->screen_tiles_ptr_offset);
|
2011-03-18 04:09:26 -06:00
|
|
|
uint8_t* tiles = new uint8_t[width*height*4/* + 80 + width*height*4*/];
|
|
|
|
|
2012-01-03 17:45:11 -07:00
|
|
|
d->owner->read (screen_addr, (width*height*4/* + 80 + width*height*4*/), tiles);
|
2011-03-18 04:09:26 -06:00
|
|
|
|
|
|
|
for(int32_t iy=0; iy<height; iy++)
|
|
|
|
{
|
|
|
|
for(int32_t ix=0; ix<width; ix++)
|
|
|
|
{
|
|
|
|
screen[ix + iy*width].symbol = tiles[(iy + ix*height)*4 +0];
|
|
|
|
screen[ix + iy*width].foreground = tiles[(iy + ix*height)*4 +1];
|
|
|
|
screen[ix + iy*width].background = tiles[(iy + ix*height)*4 +2];
|
|
|
|
screen[ix + iy*width].bright = tiles[(iy + ix*height)*4 +3];
|
|
|
|
//screen[ix + iy*width].gtile = tiles[width*height*4 + 80 + iy + ix*height +0];
|
|
|
|
//screen[ix + iy*width].grayscale = tiles[width*height*4 + 80 + iy + ix*height +1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete [] tiles;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|