问题
I'm working on Delphi XE2 application targetting Mac OS and Windows. And I want to have integration into context menu. For windows this is simple task. But for Mac OS I dont know how to do this.
I've read Providing a Service documentation and tried similar code in Delphi but with no luck.
Look at simple code for Finder integration trials.
App.dpr
program App;
uses
SysUtils,
{$IFDEF MACOS}
AppKit, CocoaTypes, CoreFoundation,
CoreServices, Foundation, Mach, ObjCRuntime,
ObjectiveC, OCMarshal, OpenGL, QuartzCore, Security,
SystemConfiguration,
{$ENDIF}
MessageProvider;
{$IFDEF MACOS}
var
app: NSApplication;
provider: TMessageProvider;
{$ENDIF}
begin
Application.Initialize;
{$IFDEF MACOS}
provider := TMessageProvider.Create();
app := TNSApplication.Alloc();
app.setServicesProvider(provider);
{$ENDIF}
Application.CreateForm(TFormOSVersion, FormOSVersion);
Application.Run;
end.
MessageProvider.pas
unit MessageProvider;
interface
uses
FMX.Dialogs
{$IFDEF MACOS}
, AppKit, CocoaTypes, CoreFoundation,
CoreServices, Foundation, Mach, ObjCRuntime,
ObjectiveC, OCMarshal, OpenGL, QuartzCore, Security,
SystemConfiguration
{$ENDIF}
;
type
TMessageProvider = class
public
procedure simpleMessage(var userData: string; var error: string);
end;
implementation
procedure TMessageProvider.simpleMessage(var userData: string; var error: string);
begin
ShowMessage('Simple message from service.');
error := '';
end;
end.
Added configuration to info.plist
<key>NSServices</key>
<array>
<dict>
<key>NSKeyEquivalent</key>
<dict>
<key>default</key>
<string>e</string>
</dict>
<key>NSMenuItem</key>
<dict>
<key>default</key>
<string>App/Message</string>
</dict>
<key>NSMessage</key>
<string>simpleMesage</string>
<key>NSPortName</key>
<string>App</string>
</dict>
</array>
When run this on Mac OS application hungs and sometimes crashes with 'Bus error' exception.
Can anybody help with this problem?
Or maybe Delphi XE2 doesnt support this kind of functionality?
回答1:
Finally, I returned to this project and successfully registered service provider and handled service request.
First of all I tried to use NSRegisterServicesProvider method, but there is no such method in Macapi sources, so I searched for applicationDidFinishLaunching delegate. Using it I registered my service provider:
procedure TApplicationDelegate.applicationDidFinishLaunching(Notification: Pointer);
var
autoReleasePool: NSAutoreleasePool;
app: NSApplication;
provider: TMessageProvider;
begin
autoReleasePool := TNSAutoreleasePool.Create;
try
autoReleasePool.init();
app := TNSApplication.Wrap(TNSApplication.OCClass.sharedApplication);
provider := TMessageProvider.Create();
app.setServicesProvider(provider.ObjId);
finally
autoReleasePool.release();
end;
end;
Also I have created interface for service provider (I think it is required for ObjectiveC-Delphi bridge work):
IMessageProvider = interface(IObjectiveC)['{1EA9319A-8F99-4445-B435-48D5E73876FA}']
procedure simpleMessage(pBoard: Pointer; userData: Pointer; error: PPointer); cdecl;
end;
and inherited TMessageProvider from this interface and TOCLocal class.
After this my app can react to service request from context menu.
I've shared sources of my project. Here they are.
回答2:
I see two potential problems
You are allocating your own
NSApplication
object. I doubt that this is correct - doesn't Delphi create one internally also? And even if it doesn't, you'd probably need to enter theNSApplication
'srun
method at some point to make it actually capable of handling messages.Service providers must be registeres in the
applicationDidFinishLaunching:
delegate method. You attempt to register it immediatly after creating yourNSApplication
instance.
I think you can avoid both problems if you use NSRegisterServicesProvider(id provider, NSString *portName)
to register your service provide, instead of using NSApplication
's setServicesProvider:
.
来源:https://stackoverflow.com/questions/8013096/howto-add-menu-item-to-mac-os-finder-in-delphi-xe2