2012-08-02 12:18:38 +01:00
|
|
|
// NasalCanvas.cxx -- expose Canvas classes to Nasal
|
|
|
|
//
|
|
|
|
// Written by James Turner, started 2012.
|
|
|
|
//
|
|
|
|
// Copyright (C) 2012 James Turner
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License as
|
|
|
|
// published by the Free Software Foundation; either version 2 of the
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful, but
|
|
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
#include <memory>
|
2012-08-02 12:18:38 +01:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "NasalCanvas.hxx"
|
2012-11-09 00:06:17 +01:00
|
|
|
#include <Canvas/canvas_mgr.hxx>
|
|
|
|
#include <Main/globals.hxx>
|
2012-08-02 12:18:38 +01:00
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
//#include <boost/python.hpp>
|
2012-08-02 12:18:38 +01:00
|
|
|
#include <boost/foreach.hpp>
|
|
|
|
#include <boost/algorithm/string/case_conv.hpp>
|
2012-11-09 00:06:17 +01:00
|
|
|
#include <boost/make_shared.hpp>
|
2012-08-02 12:18:38 +01:00
|
|
|
#include <osgGA/GUIEventAdapter>
|
|
|
|
|
|
|
|
#include <simgear/sg_inlines.h>
|
|
|
|
|
2012-11-04 14:18:31 +01:00
|
|
|
#include <simgear/canvas/Canvas.hxx>
|
2012-11-04 23:03:06 +01:00
|
|
|
#include <simgear/canvas/elements/CanvasElement.hxx>
|
2012-08-02 12:18:38 +01:00
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
#include <simgear/nasal/cppbind/from_nasal.hxx>
|
2012-11-09 19:33:03 +01:00
|
|
|
#include <simgear/nasal/cppbind/to_nasal.hxx>
|
|
|
|
#include <simgear/nasal/cppbind/NasalHash.hxx>
|
2012-11-12 12:13:06 +01:00
|
|
|
#include <simgear/nasal/cppbind/Ghost.hxx>
|
2012-08-02 12:18:38 +01:00
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
extern naRef propNodeGhostCreate(naContext c, SGPropertyNode* n);
|
2012-08-02 12:18:38 +01:00
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
//void initCanvasPython()
|
|
|
|
//{
|
|
|
|
// using namespace boost::python;
|
|
|
|
// class_<simgear::canvas::Canvas>("Canvas");
|
|
|
|
//}
|
|
|
|
|
|
|
|
namespace sc = simgear::canvas;
|
|
|
|
|
2012-11-09 19:33:03 +01:00
|
|
|
naRef canvasGetNode(naContext c, sc::Canvas* canvas)
|
2012-08-02 12:18:38 +01:00
|
|
|
{
|
2012-11-09 19:33:03 +01:00
|
|
|
return propNodeGhostCreate(c, canvas->getProps());
|
2012-08-02 12:18:38 +01:00
|
|
|
}
|
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
struct Base
|
2012-08-02 12:18:38 +01:00
|
|
|
{
|
2012-11-09 00:06:17 +01:00
|
|
|
int getInt() const
|
|
|
|
{
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Test:
|
|
|
|
public Base
|
|
|
|
{
|
|
|
|
Test(): x(1) {}
|
|
|
|
int x;
|
2012-11-12 12:13:06 +01:00
|
|
|
void setX(int x_) { x = x_; }
|
|
|
|
naRef test(int argc, naRef* args)
|
|
|
|
{
|
|
|
|
return naNil();
|
|
|
|
}
|
2012-11-09 00:06:17 +01:00
|
|
|
};
|
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
typedef nasal::Ghost<sc::CanvasPtr> NasalCanvas;
|
2012-11-09 00:06:17 +01:00
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
void initCanvas(naContext c)
|
2012-11-09 00:06:17 +01:00
|
|
|
{
|
|
|
|
|
|
|
|
NasalCanvas::init("Canvas")
|
2012-11-09 19:33:03 +01:00
|
|
|
.member("_node_ghost", &canvasGetNode)
|
|
|
|
.member("size_x", &sc::Canvas::getSizeX)
|
|
|
|
.member("size_y", &sc::Canvas::getSizeY);
|
2012-11-12 12:13:06 +01:00
|
|
|
nasal::Ghost<sc::ElementPtr>::init("canvas.Element");
|
|
|
|
nasal::Ghost<sc::GroupPtr>::init("canvas.Group")
|
2012-11-09 00:06:17 +01:00
|
|
|
.bases<sc::ElementPtr>();
|
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
nasal::Ghost<Base>::init("BaseClass")
|
2012-11-09 19:33:03 +01:00
|
|
|
.member("int", &Base::getInt);
|
2012-11-12 12:13:06 +01:00
|
|
|
nasal::Ghost<Test>::init("TestClass")
|
|
|
|
.bases<Base>()
|
|
|
|
.member("x", &Test::setX)
|
|
|
|
.method<&Test::test>("test");
|
2012-08-02 12:18:38 +01:00
|
|
|
}
|
|
|
|
|
2012-11-09 19:33:03 +01:00
|
|
|
#if 0
|
2012-11-09 00:06:17 +01:00
|
|
|
/**
|
|
|
|
* Class for exposing C++ objects to Nasal
|
|
|
|
*/
|
|
|
|
template<class T, class Derived>
|
|
|
|
class NasalObject
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// TODO use variadic template when supporting C++11
|
|
|
|
template<class A1>
|
|
|
|
static naRef create( naContext c, const A1& a1 )
|
|
|
|
{
|
|
|
|
return makeGhost(c, new T(a1));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class A1, class A2>
|
|
|
|
static naRef create( naContext c, const A1& a1,
|
|
|
|
const A2& a2 )
|
|
|
|
{
|
|
|
|
return makeGhost(c, new T(a1, a2));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class A1, class A2, class A3>
|
|
|
|
static naRef create( naContext c, const A1& a1,
|
|
|
|
const A2& a2,
|
|
|
|
const A3& a3 )
|
|
|
|
{
|
|
|
|
return makeGhost(c, new T(a1, a2, a3));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class A1, class A2, class A3, class A4>
|
|
|
|
static naRef create( naContext c, const A1& a1,
|
|
|
|
const A2& a2,
|
|
|
|
const A3& a3,
|
|
|
|
const A4& a4 )
|
|
|
|
{
|
|
|
|
return makeGhost(c, new T(a1, a2, a3, a4));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class A1, class A2, class A3, class A4, class A5>
|
|
|
|
static naRef create( naContext c, const A1& a1,
|
|
|
|
const A2& a2,
|
|
|
|
const A3& a3,
|
|
|
|
const A4& a4,
|
|
|
|
const A5& a5 )
|
|
|
|
{
|
|
|
|
return makeGhost(c, new T(a1, a2, a3, a4, a5));
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO If you need more arguments just do some copy&paste :)
|
|
|
|
|
|
|
|
static Derived& getInstance()
|
|
|
|
{
|
|
|
|
static Derived instance;
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setParent(const naRef& parent)
|
|
|
|
{
|
|
|
|
// TODO check if we need to take care of reference counting/gc
|
|
|
|
_parents.resize(1);
|
|
|
|
_parents[0] = parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
|
|
|
// TODO switch to boost::/std::function (with C++11 lambdas this can make
|
|
|
|
// adding setters easier and shorter)
|
|
|
|
typedef naRef (Derived::*getter_t)(naContext, const T&);
|
2012-11-12 12:13:06 +01:00
|
|
|
typedef std::map<std::string, getter_t> MemberMap;
|
2012-11-09 00:06:17 +01:00
|
|
|
|
|
|
|
const std::string _ghost_name;
|
|
|
|
std::vector<naRef> _parents;
|
2012-11-12 12:13:06 +01:00
|
|
|
MemberMap _members;
|
2012-11-09 00:06:17 +01:00
|
|
|
|
|
|
|
NasalObject(const std::string& ghost_name):
|
|
|
|
_ghost_name( ghost_name )
|
|
|
|
{
|
|
|
|
_ghost_type.destroy = &destroyGhost;
|
|
|
|
_ghost_type.name = _ghost_name.c_str();
|
|
|
|
_ghost_type.get_member = &Derived::getMember;
|
|
|
|
_ghost_type.set_member = 0;
|
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
_members["parents"] = &NasalObject::getParents;
|
2012-11-09 00:06:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
naRef getParents(naContext c, const T&)
|
|
|
|
{
|
|
|
|
naRef parents = naNewVector(c);
|
|
|
|
for(size_t i = 0; i < _parents.size(); ++i)
|
|
|
|
naVec_append(parents, _parents[i]);
|
|
|
|
return parents;
|
|
|
|
}
|
|
|
|
|
|
|
|
static naRef makeGhost(naContext c, void *ptr)
|
|
|
|
{
|
|
|
|
std::cout << "create " << ptr << std::endl;
|
|
|
|
return naNewGhost2(c, &(getInstance()._ghost_type), ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void destroyGhost(void *ptr)
|
|
|
|
{
|
|
|
|
std::cout << "destroy " << ptr << std::endl;
|
|
|
|
delete (T*)ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* getMember(naContext c, void* g, naRef field, naRef* out)
|
|
|
|
{
|
2012-11-12 12:13:06 +01:00
|
|
|
typename MemberMap::iterator getter =
|
|
|
|
getInstance()._members.find(naStr_data(field));
|
2012-11-09 00:06:17 +01:00
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
if( getter == getInstance()._members.end() )
|
2012-11-09 00:06:17 +01:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
*out = (getInstance().*getter->second)(c, *static_cast<T*>(g));
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
naGhostType _ghost_type;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef osg::ref_ptr<osgGA::GUIEventAdapter> GUIEventPtr;
|
|
|
|
|
|
|
|
class NasalCanvasEvent:
|
|
|
|
public NasalObject<GUIEventPtr, NasalCanvasEvent>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
NasalCanvasEvent():
|
|
|
|
NasalObject("CanvasEvent")
|
|
|
|
{
|
2012-11-12 12:13:06 +01:00
|
|
|
_members["type"] = &NasalCanvasEvent::getEventType;
|
2012-11-09 00:06:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
naRef getEventType(naContext c, const GUIEventPtr& event)
|
|
|
|
{
|
|
|
|
#define RET_EVENT_STR(type, str)\
|
|
|
|
case osgGA::GUIEventAdapter::type:\
|
2012-11-09 19:33:03 +01:00
|
|
|
return nasal::to_nasal(c, str);
|
2012-11-09 00:06:17 +01:00
|
|
|
|
|
|
|
switch( event->getEventType() )
|
|
|
|
{
|
|
|
|
RET_EVENT_STR(PUSH, "push");
|
|
|
|
RET_EVENT_STR(RELEASE, "release");
|
|
|
|
RET_EVENT_STR(DOUBLECLICK, "double-click");
|
|
|
|
RET_EVENT_STR(DRAG, "drag");
|
|
|
|
RET_EVENT_STR(MOVE, "move");
|
|
|
|
RET_EVENT_STR(SCROLL, "scroll");
|
|
|
|
RET_EVENT_STR(KEYUP, "key-up");
|
|
|
|
RET_EVENT_STR(KEYDOWN, "key-down");
|
|
|
|
|
|
|
|
#undef RET_EVENT_STR
|
|
|
|
|
|
|
|
default:
|
|
|
|
return naNil();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2012-11-09 19:33:03 +01:00
|
|
|
#endif
|
|
|
|
#if 0
|
2012-08-02 12:18:38 +01:00
|
|
|
static const char* eventGhostGetMember(naContext c, void* g, naRef field, naRef* out)
|
|
|
|
{
|
|
|
|
const char* fieldName = naStr_data(field);
|
|
|
|
osgGA::GUIEventAdapter* gea = (osgGA::GUIEventAdapter*) g;
|
2012-11-09 00:06:17 +01:00
|
|
|
|
|
|
|
if (!strcmp(fieldName, "windowX")) *out = naNum(gea->getWindowX());
|
2012-08-02 12:18:38 +01:00
|
|
|
else if (!strcmp(fieldName, "windowY")) *out = naNum(gea->getWindowY());
|
|
|
|
else if (!strcmp(fieldName, "time")) *out = naNum(gea->getTime());
|
|
|
|
else if (!strcmp(fieldName, "button")) *out = naNum(gea->getButton());
|
|
|
|
else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
static naRef f_element_addButtonCallback(naContext c, naRef me, int argc, naRef* args)
|
|
|
|
{
|
2012-11-04 14:18:31 +01:00
|
|
|
simgear::canvas::Element* e = elementGhost(me);
|
2012-08-02 12:18:38 +01:00
|
|
|
if (!e) {
|
|
|
|
naRuntimeError(c, "element.addButtonCallback called on non-canvas-element object");
|
|
|
|
}
|
|
|
|
|
|
|
|
return naNil();
|
|
|
|
}
|
|
|
|
|
|
|
|
static naRef f_element_addDragCallback(naContext c, naRef me, int argc, naRef* args)
|
|
|
|
{
|
2012-11-04 14:18:31 +01:00
|
|
|
simgear::canvas::Element* e = elementGhost(me);
|
2012-08-02 12:18:38 +01:00
|
|
|
if (!e) {
|
|
|
|
naRuntimeError(c, "element.addDragCallback called on non-canvas-element object");
|
|
|
|
}
|
|
|
|
|
|
|
|
return naNil();
|
|
|
|
}
|
|
|
|
|
|
|
|
static naRef f_element_addMoveCallback(naContext c, naRef me, int argc, naRef* args)
|
|
|
|
{
|
2012-11-04 14:18:31 +01:00
|
|
|
simgear::canvas::Element* e = elementGhost(me);
|
2012-08-02 12:18:38 +01:00
|
|
|
if (!e) {
|
|
|
|
naRuntimeError(c, "element.addMoveCallback called on non-canvas-element object");
|
|
|
|
}
|
|
|
|
|
|
|
|
return naNil();
|
|
|
|
}
|
|
|
|
|
|
|
|
static naRef f_element_addScrollCallback(naContext c, naRef me, int argc, naRef* args)
|
|
|
|
{
|
2012-11-04 14:18:31 +01:00
|
|
|
simgear::canvas::Element* e = elementGhost(me);
|
2012-08-02 12:18:38 +01:00
|
|
|
if (!e) {
|
|
|
|
naRuntimeError(c, "element.addScrollCallback called on non-canvas-element object");
|
|
|
|
}
|
|
|
|
|
|
|
|
return naNil();
|
|
|
|
}
|
2012-11-09 19:33:03 +01:00
|
|
|
#endif
|
2012-08-02 12:18:38 +01:00
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
static naRef f_createCanvas(naContext c, naRef me, int argc, naRef* args)
|
2012-08-02 12:18:38 +01:00
|
|
|
{
|
2012-11-09 00:06:17 +01:00
|
|
|
std::cout << "f_createCanvas" << std::endl;
|
|
|
|
|
|
|
|
CanvasMgr* canvas_mgr =
|
|
|
|
static_cast<CanvasMgr*>(globals->get_subsystem("Canvas"));
|
|
|
|
if( !canvas_mgr )
|
|
|
|
return naNil();
|
|
|
|
|
|
|
|
return NasalCanvas::create(c, canvas_mgr->createCanvas());
|
2012-08-02 12:18:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
naRef initNasalCanvas(naRef globals, naContext c, naRef gcSave)
|
|
|
|
{
|
2012-11-09 00:06:17 +01:00
|
|
|
/*naNewHash(c);
|
2012-08-02 12:18:38 +01:00
|
|
|
hashset(c, gcSave, "canvasProto", canvasPrototype);
|
|
|
|
|
2012-11-09 00:06:17 +01:00
|
|
|
hashset(c, canvasPrototype, "getElement", naNewFunc(c, naNewCCode(c, f_canvas_getElement)));*/
|
2012-08-02 12:18:38 +01:00
|
|
|
// set any event methods
|
|
|
|
|
2012-11-09 19:33:03 +01:00
|
|
|
#if 0
|
2012-08-02 12:18:38 +01:00
|
|
|
elementPrototype = naNewHash(c);
|
|
|
|
hashset(c, gcSave, "elementProto", elementPrototype);
|
|
|
|
|
|
|
|
hashset(c, elementPrototype, "addButtonCallback", naNewFunc(c, naNewCCode(c, f_element_addButtonCallback)));
|
|
|
|
hashset(c, elementPrototype, "addDragCallback", naNewFunc(c, naNewCCode(c, f_element_addDragCallback)));
|
|
|
|
hashset(c, elementPrototype, "addMoveCallback", naNewFunc(c, naNewCCode(c, f_element_addMoveCallback)));
|
|
|
|
hashset(c, elementPrototype, "addScrollCallback", naNewFunc(c, naNewCCode(c, f_element_addScrollCallback)));
|
2012-11-09 19:33:03 +01:00
|
|
|
#endif
|
|
|
|
nasal::Hash globals_module(globals, c),
|
|
|
|
canvas_module = globals_module.createHash("canvas");
|
2012-11-09 00:06:17 +01:00
|
|
|
|
|
|
|
canvas_module.set("_new", f_createCanvas);
|
2012-11-12 12:13:06 +01:00
|
|
|
canvas_module.set("testClass", nasal::Ghost<Test>::f_create);
|
2012-11-09 00:06:17 +01:00
|
|
|
|
2012-11-12 12:13:06 +01:00
|
|
|
initCanvas(c);
|
2012-11-09 00:06:17 +01:00
|
|
|
|
2012-08-02 12:18:38 +01:00
|
|
|
return naNil();
|
|
|
|
}
|