311 lines
9.1 KiB
C++
311 lines
9.1 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE library.
|
|
Copyright (c) 2017 - ROLI Ltd.
|
|
|
|
JUCE is an open source library subject to commercial or open-source
|
|
licensing.
|
|
|
|
The code included in this file is provided under the terms of the ISC license
|
|
http://www.isc.org/downloads/software-support-policy/isc-license. Permission
|
|
To use, copy, modify, and/or distribute this software for any purpose with or
|
|
without fee is hereby granted provided that the above copyright notice and
|
|
this permission notice appear in all copies.
|
|
|
|
JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
|
|
EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
|
|
DISCLAIMED.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
namespace juce
|
|
{
|
|
|
|
//==============================================================================
|
|
namespace Android
|
|
{
|
|
class Runnable : public juce::AndroidInterfaceImplementer
|
|
{
|
|
public:
|
|
virtual void run() = 0;
|
|
|
|
private:
|
|
jobject invoke (jobject proxy, jobject method, jobjectArray args) override
|
|
{
|
|
auto* env = getEnv();
|
|
auto methodName = juce::juceString ((jstring) env->CallObjectMethod (method, JavaMethod.getName));
|
|
|
|
if (methodName == "run")
|
|
{
|
|
run();
|
|
return nullptr;
|
|
}
|
|
|
|
// invoke base class
|
|
return AndroidInterfaceImplementer::invoke (proxy, method, args);
|
|
}
|
|
};
|
|
|
|
struct Handler
|
|
{
|
|
Handler() : nativeHandler (LocalRef<jobject> (getEnv()->NewObject (AndroidHandler, AndroidHandler.constructor))) {}
|
|
~Handler() { clearSingletonInstance(); }
|
|
|
|
JUCE_DECLARE_SINGLETON (Handler, false)
|
|
|
|
bool post (jobject runnable)
|
|
{
|
|
return (getEnv()->CallBooleanMethod (nativeHandler.get(), AndroidHandler.post, runnable) != 0);
|
|
}
|
|
|
|
GlobalRef nativeHandler;
|
|
};
|
|
|
|
JUCE_IMPLEMENT_SINGLETON (Handler)
|
|
}
|
|
|
|
//==============================================================================
|
|
struct AndroidMessageQueue : private Android::Runnable
|
|
{
|
|
JUCE_DECLARE_SINGLETON_SINGLETHREADED (AndroidMessageQueue, true)
|
|
|
|
AndroidMessageQueue()
|
|
: self (CreateJavaInterface (this, "java/lang/Runnable"))
|
|
{
|
|
}
|
|
|
|
~AndroidMessageQueue()
|
|
{
|
|
JUCE_ASSERT_MESSAGE_THREAD
|
|
clearSingletonInstance();
|
|
}
|
|
|
|
bool post (MessageManager::MessageBase::Ptr&& message)
|
|
{
|
|
queue.add (std::move (message));
|
|
|
|
// this will call us on the message thread
|
|
return handler.post (self.get());
|
|
}
|
|
|
|
private:
|
|
|
|
void run() override
|
|
{
|
|
for (;;)
|
|
{
|
|
MessageManager::MessageBase::Ptr message (queue.removeAndReturn (0));
|
|
|
|
if (message == nullptr)
|
|
break;
|
|
|
|
message->messageCallback();
|
|
}
|
|
}
|
|
|
|
// the this pointer to this class in Java land
|
|
GlobalRef self;
|
|
|
|
ReferenceCountedArray<MessageManager::MessageBase, CriticalSection> queue;
|
|
Android::Handler handler;
|
|
};
|
|
|
|
JUCE_IMPLEMENT_SINGLETON (AndroidMessageQueue)
|
|
|
|
//==============================================================================
|
|
void MessageManager::doPlatformSpecificInitialisation() { AndroidMessageQueue::getInstance(); }
|
|
void MessageManager::doPlatformSpecificShutdown() { AndroidMessageQueue::deleteInstance(); }
|
|
|
|
//==============================================================================
|
|
bool MessageManager::dispatchNextMessageOnSystemQueue (const bool)
|
|
{
|
|
Logger::outputDebugString ("*** Modal loops are not possible in Android!! Exiting...");
|
|
exit (1);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool MessageManager::postMessageToSystemQueue (MessageManager::MessageBase* const message)
|
|
{
|
|
return AndroidMessageQueue::getInstance()->post (message);
|
|
}
|
|
|
|
//==============================================================================
|
|
void MessageManager::broadcastMessage (const String&)
|
|
{
|
|
}
|
|
|
|
void MessageManager::runDispatchLoop()
|
|
{
|
|
}
|
|
|
|
void MessageManager::stopDispatchLoop()
|
|
{
|
|
struct QuitCallback : public CallbackMessage
|
|
{
|
|
QuitCallback() {}
|
|
|
|
void messageCallback() override
|
|
{
|
|
auto* env = getEnv();
|
|
LocalRef<jobject> activity (getCurrentActivity());
|
|
|
|
if (activity != nullptr)
|
|
{
|
|
jmethodID quitMethod = env->GetMethodID (AndroidActivity, "finishAndRemoveTask", "()V");
|
|
|
|
if (quitMethod != 0)
|
|
{
|
|
env->CallVoidMethod (activity.get(), quitMethod);
|
|
return;
|
|
}
|
|
|
|
quitMethod = env->GetMethodID (AndroidActivity, "finish", "()V");
|
|
jassert (quitMethod != 0);
|
|
env->CallVoidMethod (activity.get(), quitMethod);
|
|
}
|
|
else
|
|
{
|
|
jassertfalse;
|
|
}
|
|
}
|
|
};
|
|
|
|
(new QuitCallback())->post();
|
|
quitMessagePosted = true;
|
|
}
|
|
|
|
//==============================================================================
|
|
class JuceAppLifecycle : public ActivityLifecycleCallbacks
|
|
{
|
|
public:
|
|
JuceAppLifecycle (juce::JUCEApplicationBase* (*initSymbolAddr)())
|
|
: createApplicationSymbol (initSymbolAddr)
|
|
{
|
|
LocalRef<jobject> appContext (getAppContext());
|
|
|
|
if (appContext != nullptr)
|
|
{
|
|
auto* env = getEnv();
|
|
|
|
myself = GlobalRef (CreateJavaInterface (this, "android/app/Application$ActivityLifecycleCallbacks"));
|
|
env->CallVoidMethod (appContext.get(), AndroidApplication.registerActivityLifecycleCallbacks, myself.get());
|
|
}
|
|
}
|
|
|
|
~JuceAppLifecycle()
|
|
{
|
|
LocalRef<jobject> appContext (getAppContext());
|
|
|
|
if (appContext != nullptr && myself != nullptr)
|
|
{
|
|
auto* env = getEnv();
|
|
|
|
clear();
|
|
env->CallVoidMethod (appContext.get(), AndroidApplication.unregisterActivityLifecycleCallbacks, myself.get());
|
|
myself.clear();
|
|
}
|
|
}
|
|
|
|
void onActivityCreated (jobject, jobject) override
|
|
{
|
|
checkCreated();
|
|
}
|
|
|
|
void onActivityDestroyed (jobject activity) override
|
|
{
|
|
auto* env = getEnv();
|
|
|
|
// if the main activity is being destroyed, only then tear-down JUCE
|
|
if (env->IsSameObject (getMainActivity().get(), activity) != 0)
|
|
{
|
|
JUCEApplicationBase::appWillTerminateByForce();
|
|
JNIClassBase::releaseAllClasses (env);
|
|
|
|
jclass systemClass = (jclass) env->FindClass ("java/lang/System");
|
|
jmethodID exitMethod = env->GetStaticMethodID (systemClass, "exit", "(I)V");
|
|
env->CallStaticVoidMethod (systemClass, exitMethod, 0);
|
|
}
|
|
}
|
|
|
|
void onActivityStarted (jobject) override
|
|
{
|
|
checkCreated();
|
|
}
|
|
|
|
void onActivityPaused (jobject) override
|
|
{
|
|
if (auto* app = JUCEApplicationBase::getInstance())
|
|
app->suspended();
|
|
}
|
|
|
|
void onActivityResumed (jobject) override
|
|
{
|
|
checkInitialised();
|
|
|
|
if (auto* app = JUCEApplicationBase::getInstance())
|
|
app->resumed();
|
|
}
|
|
|
|
static JuceAppLifecycle& getInstance (juce::JUCEApplicationBase* (*initSymbolAddr)())
|
|
{
|
|
static JuceAppLifecycle juceAppLifecycle (initSymbolAddr);
|
|
return juceAppLifecycle;
|
|
}
|
|
|
|
private:
|
|
void checkCreated()
|
|
{
|
|
if (JUCEApplicationBase::getInstance() == nullptr)
|
|
{
|
|
DBG (SystemStats::getJUCEVersion());
|
|
|
|
JUCEApplicationBase::createInstance = createApplicationSymbol;
|
|
|
|
initialiseJuce_GUI();
|
|
|
|
if (! JUCEApplicationBase::createInstance())
|
|
jassertfalse; // you must supply an application object for an android app!
|
|
|
|
jassert (MessageManager::getInstance()->isThisTheMessageThread());
|
|
}
|
|
}
|
|
|
|
void checkInitialised()
|
|
{
|
|
checkCreated();
|
|
|
|
if (! hasBeenInitialised)
|
|
{
|
|
if (auto* app = JUCEApplicationBase::getInstance())
|
|
{
|
|
hasBeenInitialised = app->initialiseApp();
|
|
|
|
if (! hasBeenInitialised)
|
|
exit (app->shutdownApp());
|
|
}
|
|
}
|
|
}
|
|
|
|
GlobalRef myself;
|
|
juce::JUCEApplicationBase* (*createApplicationSymbol)();
|
|
bool hasBeenInitialised = false;
|
|
};
|
|
|
|
//==============================================================================
|
|
File juce_getExecutableFile();
|
|
|
|
void juce_juceEventsAndroidStartApp()
|
|
{
|
|
auto dllPath = juce_getExecutableFile().getFullPathName();
|
|
auto addr = reinterpret_cast<juce::JUCEApplicationBase*(*)()> (DynamicLibrary (dllPath)
|
|
.getFunction ("juce_CreateApplication"));
|
|
|
|
if (addr != nullptr)
|
|
JuceAppLifecycle::getInstance (addr);
|
|
}
|
|
|
|
} // namespace juce
|