gecko-dev/toolkit/xre/nsNativeAppSupportBeOS.cpp

383 lines
11 KiB
C++

/* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is the Mozilla Browser.
*
* The Initial Developer of the Original Code is
* Fredrik Holmqvist <thesuckiestemail@yahoo.se>.
* Portions created by the Initial Developer are Copyright (C) 2005
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Sergei Dolgov <sergei_d@fi.tartu.ee>
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
//This define requires DebugConsole (see BeBits.com) to be installed
//#define DC_PROGRAMNAME "firefox-bin"
# ifdef DC_PROGRAMNAME
#include <DebugConsole.h>
#endif
#include "nsIServiceManager.h"
#include "nsNativeAppSupportBase.h"
#include "nsICommandLineRunner.h"
#include "nsCOMPtr.h"
#include "nsIProxyObjectManager.h"
//#include "nsIBrowserDOMWindow.h"
#include "nsPIDOMWindow.h"
#include "nsIDOMChromeWindow.h"
#include "nsIWindowMediator.h"
#include "nsXPIDLString.h"
#include "nsIBaseWindow.h"
#include "nsIWidget.h"
#include "nsIDocShell.h"
#include <Application.h>
#include <AppFileInfo.h>
#include <Resources.h>
#include <Path.h>
#include <Window.h>
#include <unistd.h>
// Two static helpers for future - if we decide to use OpenBrowserWindow, like we do in SeaMonkey
static nsresult
GetMostRecentWindow(const PRUnichar* aType, nsIDOMWindowInternal** aWindow)
{
nsresult rv;
nsCOMPtr<nsIWindowMediator> med(do_GetService( NS_WINDOWMEDIATOR_CONTRACTID, &rv));
if (NS_FAILED(rv))
return rv;
if (med)
{
nsCOMPtr<nsIWindowMediator> medProxy;
rv = NS_GetProxyForObject(NS_PROXY_TO_MAIN_THREAD, NS_GET_IID(nsIWindowMediator),
med, NS_PROXY_SYNC | NS_PROXY_ALWAYS,
getter_AddRefs(medProxy));
if (NS_FAILED(rv))
return rv;
return medProxy->GetMostRecentWindow( aType, aWindow );
}
return NS_ERROR_FAILURE;
}
static nsresult
ActivateWindow(nsIDOMWindowInternal* aWindow)
{
nsCOMPtr<nsPIDOMWindow> window(do_QueryInterface(aWindow));
NS_ENSURE_TRUE(window, NS_ERROR_FAILURE);
nsCOMPtr<nsIBaseWindow> baseWindow(do_QueryInterface(window->GetDocShell()));
NS_ENSURE_TRUE(baseWindow, NS_ERROR_FAILURE);
nsCOMPtr<nsIWidget> mainWidget;
baseWindow->GetMainWidget(getter_AddRefs(mainWidget));
NS_ENSURE_TRUE(mainWidget, NS_ERROR_FAILURE);
BWindow *bwindow = (BWindow *)(mainWidget->GetNativeData(NS_NATIVE_WINDOW));
if (bwindow)
bwindow->Activate(true);
return NS_OK;
}
//End static helpers
class nsNativeAppSupportBeOS : public nsNativeAppSupportBase
{
public:
NS_DECL_ISUPPORTS
NS_DECL_NSINATIVEAPPSUPPORT
static void HandleCommandLine( int32 argc, char **argv, PRUint32 aState);
}; // nsNativeAppSupportBeOS
class nsBeOSApp : public BApplication
{
public:
nsBeOSApp(sem_id sem) : BApplication( GetAppSig() ), init(sem), mMessage(NULL)
{}
~nsBeOSApp()
{
delete mMessage;
}
void ReadyToRun()
{
release_sem(init);
}
static int32 Main( void *args )
{
nsBeOSApp *app = new nsBeOSApp((sem_id)args);
if (app == NULL)
return B_ERROR;
return app->Run();
}
void ArgvReceived(int32 argc, char **argv)
{
if (IsLaunching())
{
#ifdef DC_PROGRAMNAME
TRACE("ArgvReceived Launching\n");
#endif
return;
}
PRInt32 aState = /*IsLaunching() ?
nsICommandLine::STATE_INITIAL_LAUNCH :*/
nsICommandLine::STATE_REMOTE_AUTO;
nsNativeAppSupportBeOS::HandleCommandLine(argc, argv, aState);
}
void RefsReceived(BMessage* msg)
{
#ifdef DC_PROGRAMNAME
TRACE("RefsReceived\n");
#endif
if (IsLaunching())
{
mMessage = new BMessage(*msg);
return;
}
BPath path;
entry_ref er;
for (uint32 i = 0; msg->FindRef("refs", i, &er) == B_OK; i++)
{
int Argc = 2;
char **Argv = new char*[ 3 ];
BEntry entry(&er, true);
BEntry fentry(GetAppFile(), false);
entry.GetPath(&path);
Argv[0] = strdup( GetAppFile() ? GetAppFile() : "" );
Argv[1] = strdup( path.Path() ? path.Path() : "" );
// Safety measure
Argv[2] = 0;
// Is started, call ArgReceived, delete mArgv, else store for future usage
// after ::Enable() was called
ArgvReceived(2, Argv);
Argc = 0;
delete [] Argv;
Argv = NULL;
}
}
void MessageReceived(BMessage* msg)
{
// BMessage from nsNativeAppBeOS::Enable() received.
// Services are ready, so we can supply stored refs
if (msg->what == 'enbl' && mMessage)
{
#ifdef DC_PROGRAMNAME
TRACE("enbl received");
#endif
be_app_messenger.SendMessage(mMessage);
}
// Processing here file drop events from BWindow
// - until we implement native DnD in widget.
else if (msg->what == B_SIMPLE_DATA)
{
RefsReceived(msg);
}
else
BApplication::MessageReceived(msg);
}
private:
char *GetAppSig()
{
image_info info;
int32 cookie = 0;
BFile file;
BAppFileInfo appFileInfo;
static char sig[B_MIME_TYPE_LENGTH];
sig[0] = 0;
if (get_next_image_info(0, &cookie, &info) == B_OK &&
file.SetTo(info.name, B_READ_ONLY) == B_OK &&
appFileInfo.SetTo(&file) == B_OK &&
appFileInfo.GetSignature(sig) == B_OK)
return sig;
return "application/x-vnd.Mozilla";
}
char *GetAppFile()
{
image_info info;
int32 cookie = 0;
if (get_next_image_info(0, &cookie, &info) == B_OK && strlen(info.name) > 0)
return info.name;
return "";
}
sem_id init;
BMessage *mMessage;
}; //class nsBeOSApp
// Create and return an instance of class nsNativeAppSupportBeOS.
nsresult
NS_CreateNativeAppSupport(nsINativeAppSupport **aResult)
{
if (!aResult)
return NS_ERROR_NULL_POINTER;
nsNativeAppSupportBeOS *pNative = new nsNativeAppSupportBeOS;
if (!pNative)
return NS_ERROR_OUT_OF_MEMORY;
*aResult = pNative;
NS_ADDREF(*aResult);
return NS_OK;
}
NS_IMPL_ISUPPORTS1(nsNativeAppSupportBeOS, nsINativeAppSupport)
void
nsNativeAppSupportBeOS::HandleCommandLine(int32 argc, char **argv, PRUint32 aState)
{
nsresult rv;
// Here we get stuck when starting from file-click or "OpenWith".
// No cmdLine or any other service can be created
// To workaround the problem, we store arguments if IsLaunching()
// and using this after ::Enable() was called.
nsCOMPtr<nsICommandLineRunner> cmdLine(do_CreateInstance("@mozilla.org/toolkit/command-line;1"));
if (!cmdLine)
{
#ifdef DC_PROGRAMNAME
TRACE("Couldn't create command line!");
#endif
return;
}
// nsICommandLineRunner::Init() should be called from main mozilla thread
// but we are at be_app thread. Using proxy to switch thread
nsCOMPtr<nsICommandLineRunner> cmdLineProxy;
rv = NS_GetProxyForObject( NS_PROXY_TO_MAIN_THREAD, NS_GET_IID(nsICommandLineRunner),
cmdLine, NS_PROXY_ASYNC | NS_PROXY_ALWAYS, getter_AddRefs(cmdLineProxy));
if (rv != NS_OK)
{
#ifdef DC_PROGRAMNAME
TRACE("Couldn't get command line Proxy!");
#endif
return;
}
// nsICommandLineRunner::Init(,,workingdir,) requires some folder to be provided
// but that's unclear if we need it, so using 0 instead atm
rv = cmdLine->Init(argc, argv, 0 , aState);
if (rv != NS_OK)
{
#ifdef DC_PROGRAMNAME
TRACE("Couldn't init command line!");
#endif
return;
}
nsCOMPtr<nsIDOMWindowInternal> navWin;
GetMostRecentWindow( NS_LITERAL_STRING( "navigator:browser" ).get(),
getter_AddRefs(navWin ));
if (navWin)
{
# ifdef DC_PROGRAMNAME
TRACE("GotNavWin!");
# endif
cmdLine->SetWindowContext(navWin);
}
// TODO: try to use OpenURI here if there is navWin, maybe using special function
// OpenBrowserWindow which calls OpenURI like we do for SeaMonkey,
// else let CommandLineRunner to do its work.
// Problem with current implementation is unsufficient tabbed browsing support
cmdLineProxy->Run();
}
NS_IMETHODIMP
nsNativeAppSupportBeOS::Start(PRBool *aResult)
{
NS_ENSURE_ARG(aResult);
NS_ENSURE_TRUE(be_app == NULL, NS_ERROR_NOT_INITIALIZED);
sem_id initsem = create_sem(0, "Mozilla BApplication init");
if (initsem < B_OK)
return NS_ERROR_FAILURE;
thread_id tid = spawn_thread(nsBeOSApp::Main, "Mozilla XUL BApplication", B_NORMAL_PRIORITY, (void *)initsem);
#ifdef DC_PROGRAMNAME
TRACE("BeApp created");
#endif
*aResult = PR_TRUE;
if (tid < B_OK || B_OK != resume_thread(tid))
*aResult = PR_FALSE;
if (B_OK != acquire_sem(initsem))
*aResult = PR_FALSE;
if (B_OK != delete_sem(initsem))
*aResult = PR_FALSE;
return *aResult == PR_TRUE ? NS_OK : NS_ERROR_FAILURE;
}
NS_IMETHODIMP
nsNativeAppSupportBeOS::Stop(PRBool *aResult)
{
NS_ENSURE_ARG(aResult);
NS_ENSURE_TRUE(be_app, NS_ERROR_NOT_INITIALIZED);
*aResult = PR_TRUE;
return NS_OK;
}
NS_IMETHODIMP
nsNativeAppSupportBeOS::Quit()
{
if (be_app->Lock())
{
be_app->Quit();
return NS_OK;
}
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP
nsNativeAppSupportBeOS::ReOpen()
{
return NS_ERROR_NOT_IMPLEMENTED;
}
NS_IMETHODIMP
nsNativeAppSupportBeOS::Enable()
{
// Informing be_app that UI and services are ready to use.
if (be_app)
{
be_app_messenger.SendMessage('enbl');
}
return NS_OK;
}
NS_IMETHODIMP
nsNativeAppSupportBeOS::OnLastWindowClosing()
{
return NS_OK;
}