Как добавить пункт меню в Mac OS Finder в Delphi XE2 - PullRequest
31 голосов
/ 04 ноября 2011

Я работаю над приложением Delphi XE2 для Mac OS и Windows.И я хочу иметь интеграцию в контекстное меню.Для окон это простая задача.Но для Mac OS я не знаю, как это сделать.

Я прочитал Предоставление документации по обслуживанию и пробовал подобный код в Delphi, но безуспешно.

Посмотритепростой код для испытаний интеграции Finder.

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.

Добавлена ​​конфигурация в 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>

При запуске этого в Mac OS приложение зависает и иногда падает с исключением «Ошибка шины».

Кто-нибудь может помочь с этой проблемой?

Или, возможно, Delphi XE2 не поддерживает такую ​​функцию

Ответы [ 2 ]

2 голосов
/ 23 марта 2013

Наконец, я вернулся к этому проекту и успешно зарегистрировал поставщика услуг и обработал запрос на обслуживание.

Прежде всего я попытался использовать метод NSRegisterServicesProvider, но в источниках Macapi такого метода нет, поэтому я искал делегата applicationDidFinishLaunching. Используя его, я зарегистрировал своего поставщика услуг:

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;

Также я создал интерфейс для провайдера услуг (думаю, это необходимо для работы моста ObjectiveC-Delphi):

IMessageProvider = interface(IObjectiveC)['{1EA9319A-8F99-4445-B435-48D5E73876FA}']
    procedure simpleMessage(pBoard: Pointer; userData: Pointer; error: PPointer); cdecl;
end;

и унаследовал TMessageProvider от этого интерфейса и класса TOCLocal.

После этого мое приложение может реагировать на запрос на обслуживание из контекстного меню.

Я поделился источниками моего проекта. Здесь они есть.

1 голос
/ 05 октября 2012

Я вижу две потенциальные проблемы

  1. Вы выделяете свой собственный NSApplication объект. Я сомневаюсь, что это правильно - разве Delphi не создает и внутренне? И даже если этого не произойдет, вам, возможно, потребуется в какой-то момент ввести метод NSApplication run, чтобы он действительно мог обрабатывать сообщения.

  2. Поставщики услуг должны быть зарегистрированы в методе делегата applicationDidFinishLaunching:. Вы пытаетесь зарегистрировать его сразу после создания экземпляра NSApplication.

Я думаю, что вы можете избежать обеих проблем, если будете использовать NSRegisterServicesProvider(id provider, NSString *portName) для регистрации своего сервиса вместо использования NSApplication setServicesProvider:.

...