First, I spent a Saturday afternoon at Powerscourt Gardens in Wicklow, which I now realise is an excellent way to start a project.
It rained, which meant that most visitors were spending their valuable free time in the main building. We pretty much had the entire park for ourselves. It wasn't raining hard.
Powerscourt Gardens has an English garden, a Japanese garden, and a Robin-Hood style valley with cannon tower.
But be that as it may, Saturday evening and Sunday was spent learning Delphi Prism and playing StarCraft.
Delphi Prism supports Cocoa via Cocoa# and via Monobjc. Monojbc is now the preferred method. Most Cocoa classes, including GUI and other classes are exposed to .NET via wrappers provided by Monobjc. It is possible to create an NSMutableString in Prism and I assume it can be cast into a System.String but I haven't tried.
But what is perhaps most important is that Monobjc allows Prism (and any other .NET language) to use Cocoa NIB files. NIB files contain, as I understand it, archived (or "serialised") Cocoa objects like windows, buttons, text fields etc.. which can be unarchived and connected to delegates at runtime. Via Monobjc a .NET class can act as the delegate for these Cocoa objects.
Delphi Prism provides tools to read NIB files into Visual Studio and Prism and allows for somewhat easy interaction between the Windows and Mac worlds. Prism also creates a bundle out of the compiled .NET assemblies and adds all required DLLs the usual way.
After building the application in Visual Studio, the result is a bundle with a Macintosh icon (in this case the default icon for Monobjc projects in Delphi Prism):
The application uses native Cocoa windows fresh out of a NIB file. Here the text in the text box is not part of the window definition from the NIB but was added by a delegate method awakeFromNib: which is implemented in Pascal.
The text field ("textMessage") is mapped to the NIB in designable.pas, a Pascal file in the NIB folder.
The code for the actual method looks like this:
method ApplicationController.AwakeFromNib;
begin
textInfo.StringValue := 'Hello Prism Cocoa 2 Judgement Day';
end;
AwakeFromNib() is mapped to the Objective-C method awakeFromNib: automatically. I think Prism's NIB reader tool does that. Not all delegate methods are automatic like that. More about that later.
Here we have an alert box (NSRunAlertPanel) displayed when the OK button in the main window is clicked.
The method for the button event has to be added as an action in the NIB file and will then show up in designable.pas in the NIB folder. Somehow that maps the Pascal method to the action in the NIB file as Cocoa expects it.
Embarcadero say that this is a well-designed feature but I think it is magic.
Code:
[ObjectiveCMessage('PushOK:')]
method PushOK(aSender: NSObject); partial; empty;
(This is the generated code in designable.pas that maps the method to the action in the NIB file.)
method ApplicationController.PushOK(aSender: NSObject);
var
msgbox: Integer := AppKitFramework.NSRunAlertPanel('Hello Prism Cocoa 2 Judgement Day', 'Somebody pressed the OK button!', 'Really?', nil, nil, nil);
begin
if msgbox = NSPanel.NSAlertDefaultReturn then textInfo.StringValue := 'Somebody pressed the OK button just then.';
end;
(This is the code I wrote to make the alert box appear and change the text in the text field.)
The text field's contents has been changed by the method called by the button pressed.
Everything in the text field is selected. I didn't care because I was only experimenting. But I guess I have to play with a few attributes a bit more.
One common problem with Cocoa (it's by design for good reason) is that closing the last window does not quit the program. This is very useful in a Web browser or word processor but annoying when the program only has one window and is useless without that one window.
So I figured this would be a good way to test surprising delegate methods (i.e. those that are not awakeFromNib:).
I display an alert box when the red button to close the window is clicked on.
This requires two things.
First, the event has to be caught, which turns out is easy because the event has its own delegate method which I can implement somewhere. If I can manage to map my Pascal method to the delegate method the NIB expects, it should work.
Then, the event has to be cancelled in case the user decides "No" and doesn't want to quit.
This definition in the public type section of the ApplicationController.pas class file maps the Pascal function to the delegate method:
[ObjectiveCMessage('windowShouldClose:')]
function WindowShouldClose(window: NSWindow): Boolean; partial;
And this code, further down, implements the function:
function ApplicationController.WindowShouldClose(window: NSWindow): Boolean;
var
msgbox: Integer := AppKitFramework.NSRunAlertPanel('Hello Prism Cocoa 2 Judgement Day', 'Really Quit?', 'Yes', 'No', nil, nil);
begin
if msgbox = NSPanel.NSAlertDefaultReturn then Environment.Exit(0) else Result := false;
end;
Thanks for the writeup!
ReplyDeleteDo you create the NIB file on Mac OS X and then use it on Windows from Visual Studio?
Yes.
ReplyDelete