Monday, 31 August 2009

Which came first...

I'm quite enjoying the flow I'm using to write this language binding - that of implementing an example first, and then going to the modules and adding all the necessary API to allow the example to 1) compile, and 2) run properly.
In some ways it's a little like test-driven-design, where you write your tests before you write the code that the tests, test. It's certainly nice to get to that point where you run the little application and everything is finally working as it should be.

Since the module API matches so closely with the Qt API/class structure, porting the examples is generally just a case of removing semi-colons, and converting -> into .
(well, okay, I'm obviously taking liberties here, because you need to instantiate the objects more verbosely, but otherwise that's it!)

An example? Okay...

Some C++ :
QVBoxLayout *leftLayout = new QVBoxLayout;
leftLayout->addLayout(topLeftLayout);
leftLayout->addWidget(caseCheckBox);
leftLayout->addWidget(fromStartCheckBox);
leftLayout->addStretch(1);

The equivalent BlitzMax :
Local leftLayout:QVBoxLayout = New QVBoxLayout.Create()
leftLayout.addLayout(topLeftLayout)
leftLayout.addWidget(caseCheckBox)
leftLayout.addWidget(fromStartCheckBox)
leftLayout.addStretch(1)

As you can see... the two are very similar.

You may be wondering why I require the Create() method call on the object instantiation.
I know, I could easily implement the New() method on the Type and have it create an instance of Qt's QVBoxLayout, but this complicates things if ever I need to create a BlitzMax QVBoxLayout object for an already existing C++ QVBoxLayout. (writing bindings can get your head in a spin!).
So, I generally have to "constructors" for each Type. A generic Function, and a Method. The nice thing about the method too, is that you can more easily implement your own subclasses of a given Type. (see any of the examples, where the main window/gadget does exactly that!)

:-)

Wednesday, 26 August 2009

The perils of pointers

I hope you're all sitting comfortably? Good, then we'll begin...

In today's lesson, we are going to learn why writing language bindings can, on occasion, make you wish you had a normal leisure-time activity - like hunting wabbits.

If you are using a framework/toolkit in everyday C++ you aren't (usually) going to come across an issue which has caught me on several occasions, which all to do with how a pointer is perceived when you pass it down from your higher-level language. For most of the time, everything "just works", but there are times when you are passing in, say a QPaintDevice*, and things start to get just a little weird (insofar as strange error messages that you aren't expecting).
When I say "just works", I mean that you'll declare your function parameter as said pointer type, and the call down to the framework will do its thing, no questions asked.
However, you pass it into a different function, and suddenly it is reporting errors that don't quite make sense. Many hours later and you can't see anything wrong with the code.

This is where you need to take a very large step back and ponder the problem at hand.
You should ask yourself. Is the pointer I'm passing into the function *really* a QPaintDevice, or is it actually a subclass, like say, a QWidget, in which case you may actually want to be defining the parameter as a QWidget* instead.
Why does it matter sometimes and not others?
I believe it's something to do with internal offsets inside instance of the QWidget, which although is also a QPaintDevice, is really a QWidget at heart.
Normally this is not a problem, because C++ *knows* what you are passing around. So when you say, "take this pointer as a QPaintDevice", it knows that yes it can, because it sees that it is a QWidget also.
But, when your pointer is appearing suddenly into the glure code from the higher-level language, C++ can't determine what the object really is. You tell it "this is a QPaintDevice", and it makes a whole raft of assumptions.

As you may imagine, I've been here, and solved an issue I've been having.

QPainter() takes a QPaintDevice* as a parameter. This might be a QWidget, or even a QImage (which isn't a widget), or other QPaintDevice subclasses.
When I was passing in a widget of some kind, up would pop a strange error :

  • QPainter::begin: A paint device can only be painted by one painter at a time.

Which would have been fine if I'd already assigned the widget to a QPainter...

Many hours later, and some last-resort Googling, it dawns on me that I perhaps, for widgets, instead of this :

  • QPainter * bmx_qt_qpainter_create(QPaintDevice * device)

I rather have this :

  • QPainter * bmx_qt_qpainter_createwithwidget(QWidget * widget);

And you know what... the error went away...


... even though a QWidget is a subclass of QPaintDevice.


Well, I've rambled on for too long again... back to work!


Thursday, 6 August 2009

Slots & Signals for BlitzMax

One of the very cool features of Qt is its Signal/Slot mechanism. It's also known as the Observer Pattern, in the Design Patterns book.
A signal is a function which is called as the result of some kind of event or action. For example, the click of a push button. A slot is a function which you may connect to a signal, so that whenever the signal is actioned, the slot function is called.

This is all well, good and easy to implement in C++ as part of Qt, because there are build tools which are there to generate glue framework code for you.
But of course, having to run extra utilities when trying to build your BlitzMax application wouldn't be very useful. One should simply be able to knock some code together and run it.

After mulling over this obvious problem for a while, I came up with what I think is quite a clever system which works in a functionally similar way to the C++ code....
On the low-level side, every C++ signal needs to be connected to a BlitzMax utility function.
On the BlitzMax side, the base QObject holds a list of all connections to any of its own signals. Instead of connecting a callback function pointer (like we would with wxMax), we rather pass in the name of the method to connect the signal to. Using reflection we can map the method name to the actual method which we will later call.
When the signal is actioned, every method (slot) which has connected to the signal is called in turn.
It all works seamlessly :-)

The sliders example is a very good demonstration of the system in action, and even shows the ability to create your own BlitzMax-based signal method.

Although we aren't really using the Qt signal/slot framework properly - since the real signal and slot connections are hard-coded into the modules - it functions in exactly the same way, which is really all we need to worry about ;-)

Tuesday, 4 August 2009

Big, Isn't it?

On a scale thing, Qt stands up there with the likes of wxWidgets for the number of classes and methods which make up the API.

I try not to think about it too much, because it's just too scary! Better to instead focus on specific functionality one Type at a time. Which is where I think my current method of gradually building up the library of fully-functional examples is helping immensely.
Before I know it, all of the really interesting stuff will be working ;-)

I'm still juggling with the issues of being able to "connect" to objects that I have been created by Qt, rather than by me. The biggest problem of course is connecting BlitzMax to a signal, which needs to be instantiated in C++. I believe I have a good method for this when I can create my own objects, but for Qt-specific objects, I'm still wandering around in the dark.
Two options I can see for this :
  1. Rather than subclass the original classes, we instead use a "wrapper" class, which itself contains all the slots/connect glue. It would need to connect itself to the "parent" class's destructor signal in order that it could clean both itself and the BlitzMax object up. All signalling would pass through this wrapper class (one per QObject super class)
  2. For Qt-generated objects, we check if we already have a connection with them in BlitzMax. If not, we attach a "signal handler" object to it, and have it connect to the destcructor signal of the object. The signal handler would simply set up the various signal/slots as per the wrapper class in 1.
Of course, now I've written all this down, No 1 seems to be the obvious course I should have taken from the start...

:-p