1
0
Fork 0

make dialogs resizable with Ctrl-dragging

At the moment the dialog grows to North-East. Only dialogs with
the <resizable> flag set to true are resizable.
This commit is contained in:
mfranz 2009-01-10 21:58:26 +00:00 committed by Tim Moore
parent 79ad3d44b8
commit 95c5a726a7
3 changed files with 85 additions and 10 deletions

View file

@ -6,6 +6,7 @@
#include <Input/input.hxx> #include <Input/input.hxx>
#include <Scripting/NasalSys.hxx> #include <Scripting/NasalSys.hxx>
#include <Main/fg_os.hxx>
#include "dialog.hxx" #include "dialog.hxx"
#include "new_gui.hxx" #include "new_gui.hxx"
@ -87,6 +88,7 @@ struct GUIInfo
void apply_format(SGPropertyNode *); void apply_format(SGPropertyNode *);
FGDialog * dialog; FGDialog * dialog;
SGPropertyNode_ptr node;
vector <SGBinding *> bindings; vector <SGBinding *> bindings;
int key; int key;
string label, legend, text, format; string label, legend, text, format;
@ -190,7 +192,9 @@ puObject *fgPopup::getActiveInputField(puObject *object)
*/ */
int fgPopup::checkHit(int button, int updown, int x, int y) int fgPopup::checkHit(int button, int updown, int x, int y)
{ {
int result = puPopup::checkHit(button, updown, x, y); int result = 0;
if (updown != PU_DRAG)
result = puPopup::checkHit(button, updown, x, y);
if (!_draggable) if (!_draggable)
return result; return result;
@ -209,13 +213,44 @@ int fgPopup::checkHit(int button, int updown, int x, int y)
if (hit & (PUCLASS_BUTTON|PUCLASS_ONESHOT|PUCLASS_INPUT)) if (hit & (PUCLASS_BUTTON|PUCLASS_ONESHOT|PUCLASS_INPUT))
return result; return result;
int px, py; getPosition(&_dlgX, &_dlgY);
getPosition(&px, &py); getSize(&_dlgW, &_dlgH);
_modifier = fgGetKeyModifiers();
_dragging = true; _dragging = true;
_dX = px - x; _startX = x;
_dY = py - y; _startY = y;
} else if (updown == PU_DRAG && _dragging) { } else if (updown == PU_DRAG && _dragging) {
setPosition(x + _dX, y + _dY); if (_modifier & KEYMOD_CTRL) {
if (!_resizable)
return result;
int w = _dlgW + x - _startX;
int h = _dlgH + y - _startY;
GUIInfo *info = (GUIInfo *)getUserData();
if (info && info->node) {
int y = _dlgY; // + _dlgH - h;
int pw, ph;
LayoutWidget wid(info->node);
wid.calcPrefSize(&pw, &ph);
if (w < pw)
w = pw;
if (h < ph)
h = ph;
// first child is always the dialog background puFrame
getFirstChild()->setSize(w, h);
setSize(w, h);
setPosition(_dlgX, y);
wid.layout(_dlgX, y, w, h);
applySize(static_cast<puObject *>(this));
}
} else {
setPosition(x + _dlgX - _startX, y + _dlgY - _startY);
}
} else { } else {
_dragging = false; _dragging = false;
} }
@ -241,6 +276,36 @@ int fgPopup::getHitObjects(puObject *object, int x, int y)
return type; return type;
} }
void fgPopup::applySize(puObject *object)
{
// compound plib widgets use setUserData() for internal purposes, so refuse to
// descend into anything that has more bits set than the following
const int validUserData = PUCLASS_VALUE|PUCLASS_OBJECT|PUCLASS_GROUP|PUCLASS_INTERFACE
|PUCLASS_FRAME|PUCLASS_TEXT|PUCLASS_BUTTON|PUCLASS_ONESHOT|PUCLASS_INPUT
|PUCLASS_ARROW|PUCLASS_DIAL|PUCLASS_POPUP;
int type = object->getType();
if (type & PUCLASS_GROUP && !(type & ~validUserData))
for (puObject *obj = ((puGroup *)object)->getFirstChild();
obj; obj = obj->getNextObject())
applySize(obj);
GUIInfo *info = (GUIInfo *)object->getUserData();
if (!info)
return;
SGPropertyNode *n = info->node;
if (!n) {
SG_LOG(SG_GENERAL, SG_ALERT, "applySize: no props");
return;
}
int x = n->getIntValue("x");
int y = n->getIntValue("y");
int w = n->getIntValue("width", 4);
int h = n->getIntValue("height", 4);
object->setPosition(x, y);
object->setSize(w, h);
}
//////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////
@ -537,10 +602,11 @@ FGDialog::makeObject (SGPropertyNode * props, int parentWidth, int parentHeight)
if (type == "dialog") { if (type == "dialog") {
puPopup * obj; puPopup * obj;
bool draggable = props->getBoolValue("draggable", true); bool draggable = props->getBoolValue("draggable", true);
bool resizable = props->getBoolValue("resizable", false);
if (props->getBoolValue("modal", false)) if (props->getBoolValue("modal", false))
obj = new puDialogBox(x, y); obj = new puDialogBox(x, y);
else else
obj = new fgPopup(x, y, draggable); obj = new fgPopup(x, y, resizable, draggable);
setupGroup(obj, props, width, height, true); setupGroup(obj, props, width, height, true);
setColor(obj, props); setColor(obj, props);
return obj; return obj;
@ -560,6 +626,7 @@ FGDialog::makeObject (SGPropertyNode * props, int parentWidth, int parentHeight)
} else if (type == "hrule" || type == "vrule") { } else if (type == "hrule" || type == "vrule") {
puFrame * obj = new puFrame(x, y, x + width, y + height); puFrame * obj = new puFrame(x, y, x + width, y + height);
obj->setBorderThickness(0); obj->setBorderThickness(0);
setupObject(obj, props);
setColor(obj, props, BACKGROUND|FOREGROUND|HIGHLIGHT); setColor(obj, props, BACKGROUND|FOREGROUND|HIGHLIGHT);
return obj; return obj;
@ -697,6 +764,7 @@ FGDialog::setupObject (puObject * object, SGPropertyNode * props)
_info.push_back(info); _info.push_back(info);
object->setLabelPlace(PUPLACE_CENTERED_RIGHT); object->setLabelPlace(PUPLACE_CENTERED_RIGHT);
object->makeReturnDefault(props->getBoolValue("default")); object->makeReturnDefault(props->getBoolValue("default"));
info->node = props;
if (props->hasValue("legend")) { if (props->hasValue("legend")) {
info->legend = props->getStringValue("legend"); info->legend = props->getStringValue("legend");

View file

@ -175,16 +175,22 @@ private:
// //
class fgPopup : public puPopup { class fgPopup : public puPopup {
public: public:
fgPopup(int x, int y, bool d = true) : puPopup(x, y) { _dragging = false; _draggable = d;} fgPopup(int x, int y, bool r = true, bool d = true) :
puPopup(x, y), _draggable(d), _resizable(r), _dragging(false)
{}
int checkHit(int b, int up, int x, int y); int checkHit(int b, int up, int x, int y);
int checkKey(int key, int updown); int checkKey(int key, int updown);
int getHitObjects(puObject *, int x, int y); int getHitObjects(puObject *, int x, int y);
puObject *getKeyObject(puObject *, int key); puObject *getKeyObject(puObject *, int key);
puObject *getActiveInputField(puObject *); puObject *getActiveInputField(puObject *);
void applySize(puObject *);
private: private:
bool _draggable; bool _draggable;
bool _resizable;
bool _dragging; bool _dragging;
int _dX, _dY; int _modifier;
int _dlgX, _dlgY, _dlgW, _dlgH;
int _startX, _startY;
}; };

View file

@ -72,7 +72,8 @@ void LayoutWidget::calcPrefSize(int* w, int* h)
*w = *h = 17*UNIT; *w = *h = 17*UNIT;
if(getBool("vertical")) *w = 4*UNIT; if(getBool("vertical")) *w = 4*UNIT;
else *h = 4*UNIT; else *h = 4*UNIT;
} else if (isType("list") || isType("airport-list") || isType("dial")) { } else if (isType("list") || isType("airport-list")
|| isType("property-list") || isType("dial")) {
*w = *h = 12*UNIT; *w = *h = 12*UNIT;
} else if (isType("hrule")) { } else if (isType("hrule")) {
*h = 1; *h = 1;