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
|
|
|
|
/**
|
|
|
|
|
* Callback to update all property values.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
update_callback (puObject * object)
|
|
|
|
|
{
|
2002-11-08 15:24:14 +00:00
|
|
|
|
((GUIWidget *)object->getUserData())->updateProperties();
|
2002-11-07 22:59:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback to close the dialog.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
close_callback (puObject * object)
|
|
|
|
|
{
|
2002-11-08 15:24:14 +00:00
|
|
|
|
delete ((GUIWidget *)object->getUserData());
|
2002-11-07 22:59:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback to apply the property value for every field.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
apply_callback (puObject * object)
|
|
|
|
|
{
|
2002-11-08 15:24:14 +00:00
|
|
|
|
((GUIWidget *)object->getUserData())->applyProperties();
|
2002-11-07 22:59:46 +00:00
|
|
|
|
update_callback(object);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Callback to apply the property values and close the dialog.
|
|
|
|
|
*/
|
|
|
|
|
static void
|
|
|
|
|
close_apply_callback (puObject * object)
|
|
|
|
|
{
|
|
|
|
|
apply_callback(object);
|
|
|
|
|
close_callback(object);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
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-11-08 15:24:14 +00:00
|
|
|
|
delete _object;
|
2002-11-07 16:27:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::display (SGPropertyNode_ptr props)
|
2002-11-07 16:27:47 +00:00
|
|
|
|
{
|
2002-11-08 15:24:14 +00:00
|
|
|
|
if (_object != 0) {
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "This widget is already active");
|
2002-11-07 16:27:47 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
_object = makeObject(props, 1024, 768);
|
2002-11-07 16:27:47 +00:00
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
if (_object != 0) {
|
|
|
|
|
_object->reveal();
|
2002-11-07 16:27:47 +00:00
|
|
|
|
} else {
|
2002-11-08 15:24:14 +00:00
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Widget "
|
|
|
|
|
<< props->getStringValue("name", "[unnamed]")
|
2002-11-07 16:27:47 +00:00
|
|
|
|
<< " does not contain a proper GUI definition");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
|
2002-11-07 22:59:46 +00:00
|
|
|
|
void
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::applyProperties ()
|
2002-11-07 22:59:46 +00:00
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < _propertyObjects.size(); i++) {
|
|
|
|
|
puObject * object = _propertyObjects[i].object;
|
|
|
|
|
SGPropertyNode_ptr node = _propertyObjects[i].node;
|
|
|
|
|
node->setStringValue(object->getStringValue());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
2002-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::updateProperties ()
|
2002-11-07 22:59:46 +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-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
|
|
|
|
{
|
2002-11-07 22:59:46 +00:00
|
|
|
|
object->setUserData(this);
|
|
|
|
|
|
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-11-07 22:59:46 +00:00
|
|
|
|
if (props->hasValue("default-value-prop")) {
|
|
|
|
|
const char * name = props->getStringValue("default-value-prop");
|
|
|
|
|
SGPropertyNode_ptr node = fgGetNode(name, true);
|
|
|
|
|
object->setValue(node->getStringValue());
|
|
|
|
|
_propertyObjects.push_back(PropertyObject(object, node));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (props->hasValue("action")) {
|
|
|
|
|
string action = props->getStringValue("action");
|
|
|
|
|
if (action == "update")
|
|
|
|
|
object->setCallback(update_callback);
|
|
|
|
|
else if (action == "close")
|
|
|
|
|
object->setCallback(close_callback);
|
|
|
|
|
else if (action == "apply")
|
|
|
|
|
object->setCallback(apply_callback);
|
|
|
|
|
else if (action == "close-apply")
|
|
|
|
|
object->setCallback(close_apply_callback);
|
|
|
|
|
else
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Unknown GUI action " + action);
|
|
|
|
|
}
|
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-11-08 15:24:14 +00:00
|
|
|
|
GUIWidget::PropertyObject::PropertyObject (puObject * o, SGPropertyNode_ptr n)
|
2002-11-07 22:59:46 +00:00
|
|
|
|
: object(o),
|
|
|
|
|
node(n)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2002-11-08 15:24:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Implementation of NewGUI.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
NewGUI::NewGUI ()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
new GUIWidget(_widgets[name]); // PUI will delete it
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|