2002-11-07 16:27:47 +00:00
|
|
|
|
// new_gui.cxx: implementation of XML-configurable GUI support.
|
|
|
|
|
|
|
|
|
|
#include "new_gui.hxx"
|
|
|
|
|
|
|
|
|
|
#include <plib/pu.h>
|
2002-11-08 02:03:56 +00:00
|
|
|
|
#include <plib/ul.h>
|
2002-11-07 16:27:47 +00:00
|
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
SG_USING_STD(vector);
|
|
|
|
|
|
|
|
|
|
#include <simgear/misc/exception.hxx>
|
|
|
|
|
#include <Main/fg_props.hxx>
|
|
|
|
|
|
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Callbacks.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2002-11-07 22:59:46 +00:00
|
|
|
|
/**
|
2002-11-08 16:33:00 +00:00
|
|
|
|
* Action callback.
|
2002-11-07 22:59:46 +00:00
|
|
|
|
*/
|
|
|
|
|
static void
|
2002-11-08 16:33:00 +00:00
|
|
|
|
action_callback (puObject * object)
|
2002-11-07 22:59:46 +00:00
|
|
|
|
{
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIInfo * info = (GUIInfo *)object->getUserData();
|
|
|
|
|
NewGUI * gui =
|
|
|
|
|
(NewGUI *)globals->get_subsystem_mgr()
|
|
|
|
|
->get_group(FGSubsystemMgr::INIT)->get_subsystem("gui");
|
|
|
|
|
gui->setCurrentWidget(info->widget);
|
|
|
|
|
for (int i = 0; i < info->bindings.size(); i++) {
|
|
|
|
|
std::cerr << "Firing binding " << i << ": " << info->bindings[i]->getCommandName() << std::endl;
|
|
|
|
|
info->bindings[i]->fire();
|
|
|
|
|
std::cerr << "done\n";
|
|
|
|
|
}
|
|
|
|
|
std::cerr << "All bindings fired\n";
|
|
|
|
|
gui->setCurrentWidget(0);
|
2002-11-07 22:59:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2002-11-08 16:33:00 +00:00
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
2002-12-22 19:52:26 +00:00
|
|
|
|
// Implementation of GUIInfo.
|
2002-11-08 16:33:00 +00:00
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
2002-11-07 22:59:46 +00:00
|
|
|
|
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIInfo::GUIInfo (GUIWidget * w)
|
|
|
|
|
: widget(w)
|
2002-11-07 22:59:46 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIInfo::~GUIInfo ()
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < bindings.size(); i++) {
|
|
|
|
|
delete bindings[i];
|
|
|
|
|
bindings[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-07 22:59:46 +00:00
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Implementation of GUIWidget.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
2002-11-07 22:59:46 +00:00
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::GUIWidget (SGPropertyNode_ptr props)
|
|
|
|
|
: _object(0)
|
2002-11-07 16:27:47 +00:00
|
|
|
|
{
|
2002-11-08 15:24:14 +00:00
|
|
|
|
display(props);
|
2002-11-07 16:27:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::~GUIWidget ()
|
2002-11-07 16:27:47 +00:00
|
|
|
|
{
|
2002-12-22 19:52:26 +00:00
|
|
|
|
std::cerr << "Destroying widget\n";
|
2002-11-08 15:24:14 +00:00
|
|
|
|
delete _object;
|
2002-12-22 19:52:26 +00:00
|
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < _info.size(); i++) {
|
|
|
|
|
delete _info[i];
|
|
|
|
|
_info[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < _propertyObjects.size(); i++) {
|
|
|
|
|
delete _propertyObjects[i];
|
|
|
|
|
_propertyObjects[i] = 0;
|
|
|
|
|
}
|
2002-11-07 16:27:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIWidget::updateValue (const char * objectName)
|
2002-11-07 16:27:47 +00:00
|
|
|
|
{
|
2002-12-22 19:52:26 +00:00
|
|
|
|
for (int i = 0; i < _propertyObjects.size(); i++) {
|
|
|
|
|
if (_propertyObjects[i]->name == objectName)
|
|
|
|
|
_propertyObjects[i]->object
|
|
|
|
|
->setValue(_propertyObjects[i]->node->getStringValue());
|
2002-11-07 16:27:47 +00:00
|
|
|
|
}
|
2002-12-22 19:52:26 +00:00
|
|
|
|
}
|
2002-11-07 16:27:47 +00:00
|
|
|
|
|
2002-12-22 19:52:26 +00:00
|
|
|
|
void
|
|
|
|
|
GUIWidget::applyValue (const char * objectName)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < _propertyObjects.size(); i++) {
|
|
|
|
|
if (_propertyObjects[i]->name == objectName)
|
|
|
|
|
_propertyObjects[i]->node
|
|
|
|
|
->setStringValue(_propertyObjects[i]
|
|
|
|
|
->object->getStringValue());
|
2002-11-07 16:27:47 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-08 16:33:00 +00:00
|
|
|
|
void
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIWidget::updateValues ()
|
2002-11-08 16:33:00 +00:00
|
|
|
|
{
|
2002-12-22 19:52:26 +00:00
|
|
|
|
for (int i = 0; i < _propertyObjects.size(); i++) {
|
|
|
|
|
puObject * object = _propertyObjects[i]->object;
|
|
|
|
|
SGPropertyNode_ptr node = _propertyObjects[i]->node;
|
|
|
|
|
object->setValue(node->getStringValue());
|
2002-11-08 16:33:00 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2002-11-08 15:24:14 +00:00
|
|
|
|
|
2002-11-07 22:59:46 +00:00
|
|
|
|
void
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIWidget::applyValues ()
|
2002-11-07 22:59:46 +00:00
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < _propertyObjects.size(); i++) {
|
2002-12-22 19:52:26 +00:00
|
|
|
|
puObject * object = _propertyObjects[i]->object;
|
|
|
|
|
SGPropertyNode_ptr node = _propertyObjects[i]->node;
|
2002-11-07 22:59:46 +00:00
|
|
|
|
node->setStringValue(object->getStringValue());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIWidget::display (SGPropertyNode_ptr props)
|
2002-11-07 22:59:46 +00:00
|
|
|
|
{
|
2002-12-22 19:52:26 +00:00
|
|
|
|
if (_object != 0) {
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "This widget is already active");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_object = makeObject(props, 1024, 768);
|
|
|
|
|
|
|
|
|
|
if (_object != 0) {
|
|
|
|
|
_object->reveal();
|
|
|
|
|
} else {
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Widget "
|
|
|
|
|
<< props->getStringValue("name", "[unnamed]")
|
|
|
|
|
<< " does not contain a proper GUI definition");
|
2002-11-07 22:59:46 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-07 16:27:47 +00:00
|
|
|
|
puObject *
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::makeObject (SGPropertyNode * props, int parentWidth, int parentHeight)
|
2002-11-07 16:27:47 +00:00
|
|
|
|
{
|
|
|
|
|
int width = props->getIntValue("width", parentWidth);
|
|
|
|
|
int height = props->getIntValue("height", parentHeight);
|
|
|
|
|
|
|
|
|
|
int x = props->getIntValue("x", (parentWidth - width) / 2);
|
|
|
|
|
int y = props->getIntValue("y", (parentHeight - height) / 2);
|
|
|
|
|
|
|
|
|
|
string type = props->getName();
|
2002-11-08 02:03:56 +00:00
|
|
|
|
if (type == "")
|
|
|
|
|
type = props->getStringValue("type");
|
|
|
|
|
if (type == "") {
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "No type specified for GUI object");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2002-11-07 16:27:47 +00:00
|
|
|
|
|
|
|
|
|
if (type == "dialog") {
|
|
|
|
|
puPopup * dialog;
|
|
|
|
|
if (props->getBoolValue("modal", false))
|
|
|
|
|
dialog = new puDialogBox(x, y);
|
|
|
|
|
else
|
|
|
|
|
dialog = new puPopup(x, y);
|
|
|
|
|
setupGroup(dialog, props, width, height, true);
|
|
|
|
|
return dialog;
|
|
|
|
|
} else if (type == "group") {
|
|
|
|
|
puGroup * group = new puGroup(x, y);
|
|
|
|
|
setupGroup(group, props, width, height, false);
|
|
|
|
|
return group;
|
|
|
|
|
} else if (type == "input") {
|
|
|
|
|
puInput * input = new puInput(x, y, x + width, y + height);
|
|
|
|
|
setupObject(input, props);
|
|
|
|
|
return input;
|
|
|
|
|
} else if (type == "text") {
|
|
|
|
|
puText * text = new puText(x, y);
|
|
|
|
|
setupObject(text, props);
|
|
|
|
|
return text;
|
|
|
|
|
} else if (type == "button") {
|
|
|
|
|
puButton * b;
|
|
|
|
|
const char * legend = props->getStringValue("legend", "[none]");
|
|
|
|
|
if (props->getBoolValue("one-shot", true))
|
|
|
|
|
b = new puOneShot(x, y, legend);
|
|
|
|
|
else
|
|
|
|
|
b = new puButton(x, y, legend);
|
|
|
|
|
setupObject(b, props);
|
|
|
|
|
return b;
|
|
|
|
|
} else {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::setupObject (puObject * object, SGPropertyNode * props)
|
2002-11-07 16:27:47 +00:00
|
|
|
|
{
|
|
|
|
|
if (props->hasValue("legend"))
|
|
|
|
|
object->setLegend(props->getStringValue("legend"));
|
|
|
|
|
|
|
|
|
|
if (props->hasValue("label"))
|
|
|
|
|
object->setLabel(props->getStringValue("label"));
|
|
|
|
|
|
2002-12-22 19:52:26 +00:00
|
|
|
|
if (props->hasValue("property")) {
|
|
|
|
|
const char * name = props->getStringValue("name");
|
|
|
|
|
if (name == 0)
|
|
|
|
|
name = "";
|
|
|
|
|
const char * propname = props->getStringValue("property");
|
|
|
|
|
SGPropertyNode_ptr node = fgGetNode(propname, true);
|
2002-11-07 22:59:46 +00:00
|
|
|
|
object->setValue(node->getStringValue());
|
2002-12-22 19:52:26 +00:00
|
|
|
|
if (name != 0)
|
|
|
|
|
_propertyObjects.push_back(new PropertyObject(name, object, node));
|
2002-11-07 22:59:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-12-22 19:52:26 +00:00
|
|
|
|
vector<SGPropertyNode_ptr> nodes = props->getChildren("binding");
|
|
|
|
|
if (nodes.size() > 0) {
|
|
|
|
|
GUIInfo * info = new GUIInfo(this);
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < nodes.size(); i++)
|
|
|
|
|
info->bindings.push_back(new FGBinding(nodes[i]));
|
2002-11-08 16:33:00 +00:00
|
|
|
|
object->setCallback(action_callback);
|
2002-12-22 19:52:26 +00:00
|
|
|
|
object->setUserData(info);
|
|
|
|
|
_info.push_back(info);
|
2002-11-07 22:59:46 +00:00
|
|
|
|
}
|
2002-11-07 16:27:47 +00:00
|
|
|
|
|
|
|
|
|
object->makeReturnDefault(props->getBoolValue("default"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::setupGroup (puGroup * group, SGPropertyNode * props,
|
2002-11-07 16:27:47 +00:00
|
|
|
|
int width, int height, bool makeFrame)
|
|
|
|
|
{
|
|
|
|
|
setupObject(group, props);
|
|
|
|
|
|
|
|
|
|
if (makeFrame)
|
|
|
|
|
new puFrame(0, 0, width, height);
|
|
|
|
|
|
|
|
|
|
int nChildren = props->nChildren();
|
|
|
|
|
for (int i = 0; i < nChildren; i++)
|
|
|
|
|
makeObject(props->getChild(i), width, height);
|
|
|
|
|
group->close();
|
|
|
|
|
}
|
|
|
|
|
|
2002-12-22 19:52:26 +00:00
|
|
|
|
GUIWidget::PropertyObject::PropertyObject (const char * n,
|
|
|
|
|
puObject * o,
|
|
|
|
|
SGPropertyNode_ptr p)
|
|
|
|
|
: name(n),
|
|
|
|
|
object(o),
|
|
|
|
|
node(p)
|
2002-11-07 22:59:46 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Implementation of NewGUI.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
NewGUI::NewGUI ()
|
2002-12-22 19:52:26 +00:00
|
|
|
|
: _current_widget(0)
|
2002-11-08 15:24:14 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NewGUI::~NewGUI ()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
NewGUI::init ()
|
|
|
|
|
{
|
|
|
|
|
char path[1024];
|
|
|
|
|
ulMakePath(path, getenv("FG_ROOT"), "gui");
|
|
|
|
|
readDir(path);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
NewGUI::update (double delta_time_sec)
|
|
|
|
|
{
|
|
|
|
|
// NO OP
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
NewGUI::display (const string &name)
|
|
|
|
|
{
|
|
|
|
|
if (_widgets.find(name) == _widgets.end())
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Dialog " << name << " not defined");
|
|
|
|
|
else
|
2002-12-22 19:52:26 +00:00
|
|
|
|
new GUIWidget(_widgets[name]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
NewGUI::setCurrentWidget (GUIWidget * widget)
|
|
|
|
|
{
|
|
|
|
|
_current_widget = widget;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GUIWidget *
|
|
|
|
|
NewGUI::getCurrentWidget ()
|
|
|
|
|
{
|
|
|
|
|
return _current_widget;
|
2002-11-08 15:24:14 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
NewGUI::readDir (const char * path)
|
|
|
|
|
{
|
|
|
|
|
ulDir * dir = ulOpenDir(path);
|
|
|
|
|
|
|
|
|
|
if (dir == 0) {
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Failed to read GUI files from "
|
|
|
|
|
<< path);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ulDirEnt * dirEnt = ulReadDir(dir);
|
|
|
|
|
while (dirEnt != 0) {
|
|
|
|
|
char subpath[1024];
|
|
|
|
|
|
|
|
|
|
ulMakePath(subpath, path, dirEnt->d_name);
|
|
|
|
|
|
|
|
|
|
if (dirEnt->d_isdir && dirEnt->d_name[0] != '.') {
|
|
|
|
|
readDir(subpath);
|
|
|
|
|
} else {
|
|
|
|
|
SGPropertyNode_ptr props = new SGPropertyNode;
|
|
|
|
|
try {
|
|
|
|
|
readProperties(subpath, props);
|
|
|
|
|
} catch (const sg_exception &ex) {
|
|
|
|
|
SG_LOG(SG_INPUT, SG_ALERT, "Error parsing GUI file "
|
|
|
|
|
<< subpath);
|
|
|
|
|
}
|
|
|
|
|
if (!props->hasValue("name")) {
|
|
|
|
|
SG_LOG(SG_INPUT, SG_WARN, "GUI file " << subpath
|
|
|
|
|
<< " has no name; skipping.");
|
|
|
|
|
} else {
|
|
|
|
|
string name = props->getStringValue("name");
|
|
|
|
|
SG_LOG(SG_INPUT, SG_BULK, "Saving GUI node " << name);
|
|
|
|
|
_widgets[name] = props;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
dirEnt = ulReadDir(dir);
|
|
|
|
|
}
|
|
|
|
|
ulCloseDir(dir);
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-07 16:27:47 +00:00
|
|
|
|
// end of new_gui.cxx
|