|
- /*
- ==============================================================================
-
- 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.
-
- By using JUCE, you agree to the terms of both the JUCE 5 End-User License
- Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
- 27th April 2017).
-
- End User License Agreement: www.juce.com/juce-5-licence
- Privacy Policy: www.juce.com/juce-5-privacy-policy
-
- Or: You may also use this code under the terms of the GPL v3 (see
- www.gnu.org/licenses).
-
- 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
- {
-
- //==============================================================================
- // This byte-code is generated from native/java/com/roli/juce/JuceOpenGLView.java with min sdk version 16
- // See juce_core/native/java/README.txt on how to generate this byte-code.
- static const uint8 javaJuceOpenGLView[] =
- {31,139,8,8,110,106,229,91,0,3,74,117,99,101,79,112,101,110,71,76,86,105,101,119,46,100,101,120,0,109,84,63,104,19,81,24,
- 255,222,221,75,82,211,244,26,211,70,180,130,68,172,160,80,189,170,21,11,169,82,105,169,16,15,139,180,164,90,187,28,151,
- 211,92,105,239,98,114,77,139,56,20,151,58,185,40,34,34,184,56,22,167,14,213,77,165,187,179,58,56,58,43,82,7,7,127,239,79,
- 76,10,61,248,189,223,247,190,255,119,247,222,87,241,215,210,195,23,46,210,241,45,159,207,157,188,253,250,215,169,241,167,
- 83,63,159,111,216,239,78,127,114,30,230,63,252,229,68,53,34,90,43,143,228,72,63,11,208,29,38,165,79,2,159,53,255,0,24,48,
- 140,37,3,30,101,106,255,2,203,164,73,180,13,126,111,16,125,4,118,128,47,192,111,32,11,219,16,224,0,211,192,12,112,11,88,
- 0,92,32,0,238,3,15,128,199,192,19,224,37,240,22,216,1,190,2,187,64,130,171,122,16,9,34,37,116,95,41,141,132,238,191,75,
- 203,207,80,251,128,150,95,65,78,107,249,13,228,110,45,111,118,232,183,12,149,55,43,107,152,50,151,9,75,175,174,153,211,
- 220,47,235,115,105,23,107,70,178,242,227,218,143,163,211,62,189,239,215,251,188,100,131,14,105,125,235,93,196,99,104,30,
- 81,105,241,94,92,234,186,185,234,175,86,32,58,47,35,231,33,205,15,18,89,136,101,210,55,207,85,255,45,171,37,227,13,89,
- 227,40,150,131,224,168,192,104,150,102,198,225,133,180,231,80,112,20,223,76,236,107,227,221,196,111,90,242,27,48,217,207,
- 96,43,38,43,114,236,27,51,220,69,156,117,198,12,233,152,253,189,211,240,142,112,12,184,236,141,237,121,103,67,238,153,
- 222,183,229,228,88,16,6,241,21,98,37,234,43,173,120,254,116,205,15,175,57,229,192,95,61,187,232,54,93,58,226,184,97,165,
- 30,5,21,219,139,194,216,15,99,123,66,240,90,92,236,48,221,171,187,181,106,224,53,236,9,55,108,186,141,34,13,252,55,53,
- 145,201,158,89,169,223,117,61,95,100,45,210,49,199,139,150,237,122,180,20,216,139,40,104,239,173,90,36,86,38,163,92,34,
- 179,92,114,32,56,16,156,18,37,61,55,244,252,37,201,168,64,41,79,117,65,153,74,208,168,185,177,87,157,172,187,171,196,171,
- 81,35,166,116,232,198,65,211,159,173,6,13,202,69,225,213,56,118,189,170,95,153,141,230,130,176,18,173,82,94,234,132,74,
- 41,110,72,119,234,143,194,73,95,185,78,213,163,101,237,60,176,159,86,71,100,96,67,89,189,227,177,40,216,99,244,230,82,
- 214,165,233,51,56,152,41,235,58,13,49,43,101,93,222,152,167,1,82,124,2,188,113,103,12,63,128,227,240,241,245,117,190,109,
- 242,71,6,25,0,3,18,236,155,201,248,31,96,147,27,252,59,239,209,255,145,117,112,107,110,24,29,179,195,236,152,31,173,115,
- 47,102,72,130,218,115,36,73,237,89,194,10,202,38,230,9,203,182,239,178,81,80,249,197,140,49,181,143,184,31,84,80,177,35,
- 250,242,10,89,204,176,127,191,250,1,66,252,4,0,0};
-
- //==============================================================================
- struct AndroidGLCallbacks
- {
- static void attachedToWindow (JNIEnv*, jobject, jlong);
- static void detachedFromWindow (JNIEnv*, jobject, jlong);
- static void dispatchDraw (JNIEnv*, jobject, jlong, jobject);
- };
-
- //==============================================================================
- #define JNI_CLASS_MEMBERS(METHOD, STATICMETHOD, FIELD, STATICFIELD, CALLBACK) \
- METHOD (constructor, "<init>", "(Landroid/content/Context;J)V") \
- METHOD (getParent, "getParent", "()Landroid/view/ViewParent;") \
- METHOD (getHolder, "getHolder", "()Landroid/view/SurfaceHolder;") \
- METHOD (layout, "layout", "(IIII)V" ) \
- CALLBACK (AndroidGLCallbacks::attachedToWindow, "onAttchedWindowNative", "(J)V") \
- CALLBACK (AndroidGLCallbacks::detachedFromWindow, "onDetachedFromWindowNative", "(J)V") \
- CALLBACK (AndroidGLCallbacks::dispatchDraw, "onDrawNative", "(JLandroid/graphics/Canvas;)V")
-
- DECLARE_JNI_CLASS_WITH_BYTECODE (JuceOpenGLViewSurface, "com/roli/juce/JuceOpenGLView", 16, javaJuceOpenGLView, sizeof(javaJuceOpenGLView))
- #undef JNI_CLASS_MEMBERS
-
- //==============================================================================
- class OpenGLContext::NativeContext : private SurfaceHolderCallback
- {
- public:
- NativeContext (Component& comp,
- const OpenGLPixelFormat& /*pixelFormat*/,
- void* /*contextToShareWith*/,
- bool /*useMultisampling*/,
- OpenGLVersion)
- : component (comp),
- surface (EGL_NO_SURFACE), context (EGL_NO_CONTEXT)
- {
- auto env = getEnv();
-
- // Do we have a native peer that we can attach to?
- if (component.getPeer()->getNativeHandle() == nullptr)
- return;
-
- // Initialise the EGL display
- if (! initEGLDisplay())
- return;
-
- // create a native surface view
- surfaceView = GlobalRef (LocalRef<jobject>(env->NewObject (JuceOpenGLViewSurface,
- JuceOpenGLViewSurface.constructor,
- getAppContext().get(),
- reinterpret_cast<jlong> (this))));
- if (surfaceView.get() == nullptr)
- return;
-
- // add the view to the view hierarchy
- // after this the nativecontext can receive callbacks
- env->CallVoidMethod ((jobject) component.getPeer()->getNativeHandle(),
- AndroidViewGroup.addView, surfaceView.get());
-
- // initialise the geometry of the view
- auto bounds = component.getTopLevelComponent()->getLocalArea (&component, component.getLocalBounds());
- bounds *= component.getDesktopScaleFactor();
-
- updateWindowPosition (bounds);
- hasInitialised = true;
- }
-
- ~NativeContext()
- {
- auto env = getEnv();
-
- if (jobject viewParent = env->CallObjectMethod (surfaceView.get(), JuceOpenGLViewSurface.getParent))
- env->CallVoidMethod (viewParent, AndroidViewGroup.removeView, surfaceView.get());
- }
-
- //==============================================================================
- bool initialiseOnRenderThread (OpenGLContext& aContext)
- {
- jassert (hasInitialised);
-
- // has the context already attached?
- jassert (surface == EGL_NO_SURFACE && context == EGL_NO_CONTEXT);
-
- auto env = getEnv();
-
- ANativeWindow* window = nullptr;
-
- LocalRef<jobject> holder (env->CallObjectMethod (surfaceView.get(), JuceOpenGLViewSurface.getHolder));
-
- if (holder != nullptr)
- {
- LocalRef<jobject> jSurface (env->CallObjectMethod (holder.get(), AndroidSurfaceHolder.getSurface));
-
- if (jSurface != nullptr)
- {
- window = ANativeWindow_fromSurface(env, jSurface.get());
-
- // if we didn't succeed the first time, wait 25ms and try again
- if (window == nullptr)
- {
- Thread::sleep (200);
- window = ANativeWindow_fromSurface (env, jSurface.get());
- }
- }
- }
-
- if (window == nullptr)
- {
- // failed to get a pointer to the native window after second try so bail out
- jassertfalse;
- return false;
- }
-
- // create the surface
- surface = eglCreateWindowSurface (display, config, window, 0);
- jassert (surface != EGL_NO_SURFACE);
-
- ANativeWindow_release (window);
-
- // create the OpenGL context
- EGLint contextAttribs[] = { EGL_CONTEXT_CLIENT_VERSION, 2, EGL_NONE };
- context = eglCreateContext (display, config, EGL_NO_CONTEXT, contextAttribs);
- jassert (context != EGL_NO_CONTEXT);
-
- juceContext = &aContext;
- return true;
- }
-
- void shutdownOnRenderThread()
- {
- jassert (hasInitialised);
-
- // is there a context available to detach?
- jassert (surface != EGL_NO_SURFACE && context != EGL_NO_CONTEXT);
-
- eglDestroyContext (display, context);
- context = EGL_NO_CONTEXT;
-
- eglDestroySurface (display, surface);
- surface = EGL_NO_SURFACE;
- }
-
- //==============================================================================
- bool makeActive() const noexcept
- {
- if (! hasInitialised)
- return false;
-
- if (surface == EGL_NO_SURFACE || context == EGL_NO_CONTEXT)
- return false;
-
- if (! eglMakeCurrent (display, surface, surface, context))
- return false;
-
- return true;
- }
-
- bool isActive() const noexcept { return eglGetCurrentContext() == context; }
-
- static void deactivateCurrentContext()
- {
- eglMakeCurrent (display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT);
- }
-
- //==============================================================================
- void swapBuffers() const noexcept { eglSwapBuffers (display, surface); }
- bool setSwapInterval (const int) { return false; }
- int getSwapInterval() const { return 0; }
-
- //==============================================================================
- bool createdOk() const noexcept { return hasInitialised; }
- void* getRawContext() const noexcept { return surfaceView.get(); }
- GLuint getFrameBufferID() const noexcept { return 0; }
-
- //==============================================================================
- void updateWindowPosition (Rectangle<int> bounds)
- {
- if (lastBounds != bounds)
- {
- auto env = getEnv();
-
- lastBounds = bounds;
- auto r = bounds * Desktop::getInstance().getDisplays().getMainDisplay().scale;
-
- env->CallVoidMethod (surfaceView.get(), JuceOpenGLViewSurface.layout,
- (jint) r.getX(), (jint) r.getY(), (jint) r.getRight(), (jint) r.getBottom());
- }
- }
-
- //==============================================================================
- // Android Surface Callbacks:
- void surfaceChanged (LocalRef<jobject> holder, int format, int width, int height) override
- {
- ignoreUnused (holder, format, width, height);
- }
-
- void surfaceCreated (LocalRef<jobject> holder) override;
- void surfaceDestroyed (LocalRef<jobject> holder) override;
-
- //==============================================================================
- struct Locker { Locker (NativeContext&) {} };
-
- Component& component;
-
- private:
- //==============================================================================
- friend struct AndroidGLCallbacks;
-
- void attachedToWindow()
- {
- auto* env = getEnv();
-
- LocalRef<jobject> holder (env->CallObjectMethod (surfaceView.get(), JuceOpenGLViewSurface.getHolder));
-
- if (surfaceHolderCallback == nullptr)
- surfaceHolderCallback = GlobalRef (CreateJavaInterface (this, "android/view/SurfaceHolder$Callback"));
-
- env->CallVoidMethod (holder, AndroidSurfaceHolder.addCallback, surfaceHolderCallback.get());
- }
-
- void detachedFromWindow()
- {
- if (surfaceHolderCallback != nullptr)
- {
- auto* env = getEnv();
-
- LocalRef<jobject> holder (env->CallObjectMethod (surfaceView.get(), JuceOpenGLViewSurface.getHolder));
-
- env->CallVoidMethod (holder.get(), AndroidSurfaceHolder.removeCallback, surfaceHolderCallback.get());
- surfaceHolderCallback.clear();
- }
- }
-
- void dispatchDraw (jobject /*canvas*/)
- {
- if (juceContext != nullptr)
- juceContext->triggerRepaint();
- }
-
- //==============================================================================
- bool initEGLDisplay()
- {
- // already initialised?
- if (display != EGL_NO_DISPLAY)
- return true;
-
- const EGLint attribs[] =
- {
- EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
- EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
- EGL_BLUE_SIZE, 8,
- EGL_GREEN_SIZE, 8,
- EGL_RED_SIZE, 8,
- EGL_ALPHA_SIZE, 0,
- EGL_DEPTH_SIZE, 16,
- EGL_NONE
- };
-
- EGLint numConfigs;
-
- if ((display = eglGetDisplay (EGL_DEFAULT_DISPLAY)) == EGL_NO_DISPLAY)
- {
- jassertfalse;
- return false;
- }
-
- if (! eglInitialize (display, 0, 0))
- {
- jassertfalse;
- return false;
- }
-
- if (! eglChooseConfig (display, attribs, &config, 1, &numConfigs))
- {
- eglTerminate (display);
- jassertfalse;
- return false;
- }
-
- return true;
- }
-
- //==============================================================================
- bool hasInitialised = false;
-
- GlobalRef surfaceView;
- Rectangle<int> lastBounds;
-
- OpenGLContext* juceContext = nullptr;
- EGLSurface surface;
- EGLContext context;
-
- GlobalRef surfaceHolderCallback;
-
- static EGLDisplay display;
- static EGLConfig config;
-
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (NativeContext)
- };
-
- //==============================================================================
- void AndroidGLCallbacks::attachedToWindow (JNIEnv*, jobject /*this*/, jlong host)
- {
- if (auto* nativeContext = reinterpret_cast<OpenGLContext::NativeContext*> (host))
- nativeContext->attachedToWindow();
- }
-
- void AndroidGLCallbacks::detachedFromWindow (JNIEnv*, jobject /*this*/, jlong host)
- {
- if (auto* nativeContext = reinterpret_cast<OpenGLContext::NativeContext*> (host))
- nativeContext->detachedFromWindow();
- }
-
- void AndroidGLCallbacks::dispatchDraw (JNIEnv*, jobject /*this*/, jlong host, jobject canvas)
- {
- if (auto* nativeContext = reinterpret_cast<OpenGLContext::NativeContext*> (host))
- nativeContext->dispatchDraw (canvas);
- }
-
- //==============================================================================
- bool OpenGLHelpers::isContextActive()
- {
- return eglGetCurrentContext() != EGL_NO_CONTEXT;
- }
-
- } // namespace juce
|