676 lines
17 KiB
C++
676 lines
17 KiB
C++
/*
|
|
https://github.com/peterix/dfhack
|
|
Copyright (c) 2009-2011 Petr Mrázek (peterix@gmail.com)
|
|
|
|
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.
|
|
*/
|
|
|
|
|
|
#include "Internal.h"
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
#include <map>
|
|
using namespace std;
|
|
|
|
#include "modules/Gui.h"
|
|
#include "MemAccess.h"
|
|
#include "VersionInfo.h"
|
|
#include "Types.h"
|
|
#include "Error.h"
|
|
#include "ModuleFactory.h"
|
|
#include "Core.h"
|
|
#include "PluginManager.h"
|
|
#include "MiscUtils.h"
|
|
using namespace DFHack;
|
|
|
|
#include "DataDefs.h"
|
|
#include "df/world.h"
|
|
#include "df/global_objects.h"
|
|
#include "df/viewscreen_dwarfmodest.h"
|
|
#include "df/viewscreen_joblistst.h"
|
|
#include "df/viewscreen_unitlistst.h"
|
|
#include "df/viewscreen_itemst.h"
|
|
#include "df/ui_unit_view_mode.h"
|
|
#include "df/ui_sidebar_menus.h"
|
|
#include "df/ui_look_list.h"
|
|
#include "df/job.h"
|
|
#include "df/ui_build_selector.h"
|
|
#include "df/building_workshopst.h"
|
|
#include "df/building_furnacest.h"
|
|
#include "df/general_ref.h"
|
|
#include "df/unit_inventory_item.h"
|
|
#include "df/report.h"
|
|
#include "df/popup_message.h"
|
|
#include "df/interface.h"
|
|
#include "df/graphic.h"
|
|
|
|
using namespace df::enums;
|
|
using df::global::gview;
|
|
using df::global::init;
|
|
using df::global::gps;
|
|
|
|
// 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::unitjobs_hotkey(Core *, df::viewscreen *top)
|
|
{
|
|
// Require the unit or jobs list
|
|
return !!strict_virtual_cast<df::viewscreen_joblistst>(top) ||
|
|
!!strict_virtual_cast<df::viewscreen_unitlistst>(top);
|
|
}
|
|
|
|
bool DFHack::item_details_hotkey(Core *, df::viewscreen *top)
|
|
{
|
|
// Require the main dwarf mode screen
|
|
return !!strict_virtual_cast<df::viewscreen_itemst>(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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
}
|
|
|
|
bool DFHack::view_unit_hotkey(Core *c, df::viewscreen *top)
|
|
{
|
|
using df::global::ui;
|
|
using df::global::world;
|
|
using df::global::ui_selected_unit;
|
|
|
|
if (!dwarfmode_hotkey(c,top))
|
|
return false;
|
|
if (ui->main.mode != ui_sidebar_mode::ViewUnits)
|
|
return false;
|
|
if (!ui_selected_unit) // allow missing
|
|
return false;
|
|
|
|
return vector_get(world->units.other[0], *ui_selected_unit) != NULL;
|
|
}
|
|
|
|
bool DFHack::unit_inventory_hotkey(Core *c, df::viewscreen *top)
|
|
{
|
|
using df::global::ui_unit_view_mode;
|
|
|
|
if (!view_unit_hotkey(c,top))
|
|
return false;
|
|
if (!ui_unit_view_mode)
|
|
return false;
|
|
|
|
return ui_unit_view_mode->value == df::ui_unit_view_mode::Inventory;
|
|
}
|
|
|
|
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];
|
|
}
|
|
|
|
bool DFHack::any_job_hotkey(Core *c, df::viewscreen *top)
|
|
{
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_joblistst, top))
|
|
return vector_get(screen->jobs, screen->cursor_pos) != NULL;
|
|
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_unitlistst, top))
|
|
return vector_get(screen->jobs[screen->page], screen->cursor_pos[screen->page]) != NULL;
|
|
|
|
return workshop_job_hotkey(c,top);
|
|
}
|
|
|
|
df::job *DFHack::getSelectedJob(Core *c, bool quiet)
|
|
{
|
|
df::viewscreen *top = c->getTopViewscreen();
|
|
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_joblistst, top))
|
|
{
|
|
df::job *job = vector_get(screen->jobs, screen->cursor_pos);
|
|
|
|
if (!job && !quiet)
|
|
c->con.printerr("Selected unit has no job\n");
|
|
|
|
return job;
|
|
}
|
|
else if (VIRTUAL_CAST_VAR(screen, df::viewscreen_unitlistst, top))
|
|
{
|
|
int page = screen->page;
|
|
df::job *job = vector_get(screen->jobs[page], screen->cursor_pos[page]);
|
|
|
|
if (!job && !quiet)
|
|
c->con.printerr("Selected unit has no job\n");
|
|
|
|
return job;
|
|
}
|
|
else
|
|
return getSelectedWorkshopJob(c, quiet);
|
|
}
|
|
|
|
static df::unit *getAnyUnit(Core *c, df::viewscreen *top)
|
|
{
|
|
using namespace ui_sidebar_mode;
|
|
using df::global::ui;
|
|
using df::global::world;
|
|
using df::global::ui_look_cursor;
|
|
using df::global::ui_look_list;
|
|
using df::global::ui_selected_unit;
|
|
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_joblistst, top))
|
|
return vector_get(screen->units, screen->cursor_pos);
|
|
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_unitlistst, top))
|
|
return vector_get(screen->units[screen->page], screen->cursor_pos[screen->page]);
|
|
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_itemst, top))
|
|
{
|
|
df::general_ref *ref = vector_get(screen->entry_ref, screen->cursor_pos);
|
|
return ref ? ref->getUnit() : NULL;
|
|
}
|
|
|
|
if (!dwarfmode_hotkey(c,top))
|
|
return NULL;
|
|
|
|
switch (ui->main.mode) {
|
|
case ViewUnits:
|
|
{
|
|
if (!ui_selected_unit)
|
|
return NULL;
|
|
|
|
return vector_get(world->units.other[0], *ui_selected_unit);
|
|
}
|
|
case LookAround:
|
|
{
|
|
if (!ui_look_list || !ui_look_cursor)
|
|
return NULL;
|
|
|
|
auto item = vector_get(ui_look_list->items, *ui_look_cursor);
|
|
if (item && item->type == df::ui_look_list::T_items::Unit)
|
|
return item->unit;
|
|
else
|
|
return NULL;
|
|
}
|
|
default:
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
bool DFHack::any_unit_hotkey(Core *c, df::viewscreen *top)
|
|
{
|
|
return getAnyUnit(c, top) != NULL;
|
|
}
|
|
|
|
df::unit *DFHack::getSelectedUnit(Core *c, bool quiet)
|
|
{
|
|
df::unit *unit = getAnyUnit(c, c->getTopViewscreen());
|
|
|
|
if (!unit && !quiet)
|
|
c->con.printerr("No unit is selected in the UI.\n");
|
|
|
|
return unit;
|
|
}
|
|
|
|
static df::item *getAnyItem(Core *c, df::viewscreen *top)
|
|
{
|
|
using namespace ui_sidebar_mode;
|
|
using df::global::ui;
|
|
using df::global::world;
|
|
using df::global::ui_look_cursor;
|
|
using df::global::ui_look_list;
|
|
using df::global::ui_unit_view_mode;
|
|
using df::global::ui_building_item_cursor;
|
|
using df::global::ui_sidebar_menus;
|
|
|
|
if (VIRTUAL_CAST_VAR(screen, df::viewscreen_itemst, top))
|
|
{
|
|
df::general_ref *ref = vector_get(screen->entry_ref, screen->cursor_pos);
|
|
return ref ? ref->getItem() : NULL;
|
|
}
|
|
|
|
if (!dwarfmode_hotkey(c,top))
|
|
return NULL;
|
|
|
|
switch (ui->main.mode) {
|
|
case ViewUnits:
|
|
{
|
|
if (!ui_unit_view_mode || !ui_look_cursor || !ui_sidebar_menus)
|
|
return NULL;
|
|
|
|
if (ui_unit_view_mode->value != df::ui_unit_view_mode::Inventory)
|
|
return NULL;
|
|
|
|
auto inv_item = vector_get(ui_sidebar_menus->unit.inv_items, *ui_look_cursor);
|
|
return inv_item ? inv_item->item : NULL;
|
|
}
|
|
case LookAround:
|
|
{
|
|
if (!ui_look_list || !ui_look_cursor)
|
|
return NULL;
|
|
|
|
auto item = vector_get(ui_look_list->items, *ui_look_cursor);
|
|
if (item && item->type == df::ui_look_list::T_items::Item)
|
|
return item->item;
|
|
else
|
|
return NULL;
|
|
}
|
|
case BuildingItems:
|
|
{
|
|
if (!ui_building_item_cursor)
|
|
return NULL;
|
|
|
|
VIRTUAL_CAST_VAR(selected, df::building_actual, world->selected_building);
|
|
if (!selected)
|
|
return NULL;
|
|
|
|
auto inv_item = vector_get(selected->contained_items, *ui_building_item_cursor);
|
|
return inv_item ? inv_item->item : NULL;
|
|
}
|
|
default:
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
bool DFHack::any_item_hotkey(Core *c, df::viewscreen *top)
|
|
{
|
|
return getAnyItem(c, top) != NULL;
|
|
}
|
|
|
|
df::item *DFHack::getSelectedItem(Core *c, bool quiet)
|
|
{
|
|
df::item *item = getAnyItem(c, c->getTopViewscreen());
|
|
|
|
if (!item && !quiet)
|
|
c->con.printerr("No item is selected in the UI.\n");
|
|
|
|
return item;
|
|
}
|
|
|
|
//
|
|
|
|
void DFHack::showAnnouncement(std::string message, int color, bool bright)
|
|
{
|
|
using df::global::world;
|
|
using df::global::cur_year;
|
|
using df::global::cur_year_tick;
|
|
|
|
int year = 0, year_time = 0;
|
|
|
|
if (cur_year && cur_year_tick)
|
|
{
|
|
year = *cur_year;
|
|
year_time = *cur_year_tick;
|
|
}
|
|
else if (!world->status.reports.empty())
|
|
{
|
|
// Fallback: copy from the last report
|
|
df::report *last = world->status.reports.back();
|
|
year = last->year;
|
|
year_time = last->time;
|
|
}
|
|
|
|
bool continued = false;
|
|
|
|
while (!message.empty())
|
|
{
|
|
df::report *new_rep = new df::report();
|
|
|
|
new_rep->color = color;
|
|
new_rep->bright = bright;
|
|
new_rep->year = year;
|
|
new_rep->time = year_time;
|
|
|
|
new_rep->flags.bits.continuation = continued;
|
|
new_rep->flags.bits.announcement = true;
|
|
|
|
int size = std::min(message.size(), 73U);
|
|
new_rep->text = message.substr(0, size);
|
|
message = message.substr(size);
|
|
|
|
continued = true;
|
|
|
|
// Add the object to the lists
|
|
new_rep->id = world->status.next_report_id++;
|
|
|
|
world->status.reports.push_back(new_rep);
|
|
world->status.announcements.push_back(new_rep);
|
|
world->status.display_timer = 2000;
|
|
}
|
|
|
|
}
|
|
|
|
void DFHack::showPopupAnnouncement(std::string message, int color, bool bright)
|
|
{
|
|
using df::global::world;
|
|
|
|
df::popup_message *popup = new df::popup_message();
|
|
popup->text = message;
|
|
popup->color = color;
|
|
popup->bright = bright;
|
|
world->status.popups.push_back(popup);
|
|
}
|
|
|
|
//
|
|
|
|
Module* DFHack::createGui()
|
|
{
|
|
return new Gui();
|
|
}
|
|
|
|
struct Gui::Private
|
|
{
|
|
Private()
|
|
{
|
|
Started = false;
|
|
StartedMenu = false;
|
|
StartedScreen = false;
|
|
}
|
|
|
|
bool Started;
|
|
int32_t * window_x_offset;
|
|
int32_t * window_y_offset;
|
|
int32_t * window_z_offset;
|
|
|
|
bool StartedMenu;
|
|
uint8_t * menu_width_offset;
|
|
uint8_t * area_map_width_offset;
|
|
|
|
bool StartedScreen;
|
|
void * screen_tiles_ptr_offset;
|
|
|
|
Process * owner;
|
|
};
|
|
|
|
Gui::Gui()
|
|
{
|
|
Core & c = Core::getInstance();
|
|
d = new Private;
|
|
d->owner = c.p;
|
|
VersionInfo * mem = c.vinfo;
|
|
// Setting up menu state
|
|
df_menu_state = (uint32_t *) & df::global::ui->main.mode;
|
|
|
|
d->window_x_offset = (int32_t *) mem->getAddress ("window_x");
|
|
d->window_y_offset = (int32_t *) mem->getAddress ("window_y");
|
|
d->window_z_offset = (int32_t *) mem->getAddress ("window_z");
|
|
if(d->window_z_offset && d->window_y_offset && d->window_x_offset)
|
|
d->Started = true;
|
|
|
|
d->menu_width_offset = (uint8_t *) mem->getAddress("ui_menu_width");
|
|
d->area_map_width_offset = (uint8_t *) mem->getAddress("ui_area_map_width");
|
|
if (d->menu_width_offset && d->area_map_width_offset)
|
|
d->StartedMenu = true;
|
|
|
|
d->screen_tiles_ptr_offset = (void *) mem->getAddress ("screen_tiles_pointer");
|
|
if(d->screen_tiles_ptr_offset)
|
|
d->StartedScreen = true;
|
|
}
|
|
|
|
Gui::~Gui()
|
|
{
|
|
delete d;
|
|
}
|
|
|
|
bool Gui::Start()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool Gui::Finish()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
df::viewscreen * Gui::GetCurrentScreen()
|
|
{
|
|
df::viewscreen * ws = &gview->view;
|
|
while(ws)
|
|
{
|
|
if(ws->child)
|
|
ws = ws->child;
|
|
else
|
|
return ws;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
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)
|
|
{
|
|
x = df::global::cursor->x;
|
|
y = df::global::cursor->y;
|
|
z = df::global::cursor->z;
|
|
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)
|
|
{
|
|
df::global::cursor->x = x;
|
|
df::global::cursor->y = y;
|
|
df::global::cursor->z = z;
|
|
return true;
|
|
}
|
|
|
|
bool Gui::getDesignationCoords (int32_t &x, int32_t &y, int32_t &z)
|
|
{
|
|
x = df::global::selection_rect->start_x;
|
|
y = df::global::selection_rect->start_y;
|
|
z = df::global::selection_rect->start_z;
|
|
if (x == -30000) return false;
|
|
return true;
|
|
}
|
|
|
|
bool Gui::setDesignationCoords (const int32_t x, const int32_t y, const int32_t z)
|
|
{
|
|
df::global::selection_rect->start_x = x;
|
|
df::global::selection_rect->start_y = y;
|
|
df::global::selection_rect->start_z = z;
|
|
return true;
|
|
}
|
|
|
|
bool Gui::getMousePos (int32_t & x, int32_t & y)
|
|
{
|
|
x = gps->mouse_x;
|
|
y = gps->mouse_y;
|
|
if(x == -1) return false;
|
|
return true;
|
|
}
|
|
|
|
bool Gui::getWindowSize (int32_t &width, int32_t &height)
|
|
{
|
|
width = gps->dimx;
|
|
height = gps->dimy;
|
|
return true;
|
|
}
|
|
|
|
bool Gui::getMenuWidth(uint8_t &menu_width, uint8_t &area_map_width)
|
|
{
|
|
if (!d->StartedMenu) return false;
|
|
|
|
Process * p = d->owner;
|
|
|
|
p->readByte (d->menu_width_offset, menu_width);
|
|
p->readByte (d->area_map_width_offset, area_map_width);
|
|
return true;
|
|
}
|
|
|
|
bool Gui::setMenuWidth(const uint8_t menu_width, const uint8_t area_map_width)
|
|
{
|
|
if (!d->StartedMenu) return false;
|
|
|
|
Process * p = d->owner;
|
|
|
|
p->writeByte (d->menu_width_offset, menu_width);
|
|
p->writeByte (d->area_map_width_offset, area_map_width);
|
|
return true;
|
|
}
|
|
|
|
bool Gui::getScreenTiles (int32_t width, int32_t height, t_screen screen[])
|
|
{
|
|
if(!d->StartedScreen) return false;
|
|
|
|
void * screen_addr = (void *) d->owner->readDWord(d->screen_tiles_ptr_offset);
|
|
uint8_t* tiles = new uint8_t[width*height*4/* + 80 + width*height*4*/];
|
|
|
|
d->owner->read (screen_addr, (width*height*4/* + 80 + width*height*4*/), tiles);
|
|
|
|
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;
|
|
} |