0002: OOP Test Rig Breakdown
When you look at the imperative test rig alongside this OOP version, it becomes quickly apparent that the difference is mostly about code organization.
main()
The changes start in the main()
function:
void main(string[] args)
{
TestRigWindow testRigWindow;
Main.init(args);
testRigWindow = new TestRigWindow();
Main.run();
} // main()
On line 2 of main()
, the window declaration is quite different because instead of instantiating the MainWindow
class directly, there’s a derived class called TestRigWindow
. We also move any calls to the TestRigWindow
functions to the TestRigWindow
class’s constructor which strips main()
down to the bare essentials.
quitApp()
quitApp()
is no longer a standalone function as it’s been incorporated into the TestRigWindow
class which looks like this:
class TestRigWindow : MainWindow
{
string title = "Test Rig";
this()
{
super(title);
addOnDestroy(&quitApp);
sayHi();
showAll();
} // this()
void quitApp(Widget widget)
{
string exitMessage = "Bye.";
writeln(exitMessage);
Main.quit();
} // quitApp()
void sayHi()
{
string message = "Hello GtkD OOP World";
writeln(message);
} // sayHi()
} // class TestRigWindow
The first function is the constructor. D uses this()
instead of the class name (like Java) or construct()
(like PHP) and inside we:
- call
super()
, a shorthand for calling the parent class’s constructor, - while passing along the window’s title, and
- hook up the window’s close button.
Order Matters
It’s important to note here that any call to a function that will decorate the TestRigWindow
—or size it or pretty much anything else—must be made after the call to super()
. Otherwise, the window won’t exist, the function call goes off into oblivion, and we’ll get an error.
You’ll notice that connecting the onDestroy
signal is done differently here. In the imperative version, because quitApp()
is external to main()
, we have to preserve quitApp
’s scope so it doesn’t disappear on us before we can use it. But in the OOP version, the syntax is simplified because we don’t need to worry about scope. As long as the class object exists, everything within it will exist as well. This simplification of syntax makes our job easier, although it’ll get difficult again under certain circumstances. I’ll talk about that in later posts.
Another thing you’ll notice is that TestRigWindow
’s sayHi()
function has taken the place of a simple writeln()
call in the imperative version of main()
. We could have left it where it was, but this way all functions are part of the TestRigWindow
class and there’s a certain orderliness to that.
As with all examples, to compile:
dmd -de -w -m64 -Lgtkd.lib <filename>.d
Conclusion
So, now we have two test rigs and we’ve looked at how to size a window to exact specifications. That gives us a foundation to build from in the coming posts.
Until next time when we take an initial look at GUI buttons, happy D-coding and may the widgets be with you.
Comments? Questions? Observations?
Did we miss a tidbit of information that would make this post even more informative? Let's talk about it in the comments.
- come on over to the D Language Forum and look for one of the gtkDcoding announcement posts,
- drop by the GtkD Forum,
- follow the link below to email me, or
- go to the gtkDcoding Facebook page.
You can also subscribe via RSS so you won't miss anything. Thank you very much for dropping by.
© Copyright 2025 Ron Tarrant