2019-11-23 20:27:39 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2010-2012 cocos2d-x.org
|
|
|
|
Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2024-07-11 23:28:31 +08:00
|
|
|
Copyright (c) 2019-present Axmol Engine contributors (see AUTHORS.md).
|
|
|
|
|
2024-06-10 02:25:43 +08:00
|
|
|
https://axmol.dev/
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
THE SOFTWARE.
|
|
|
|
****************************************************************************/
|
2023-06-11 13:08:08 +08:00
|
|
|
#include "platform/android/GLViewImpl-android.h"
|
|
|
|
#include "base/Director.h"
|
|
|
|
#include "base/Macros.h"
|
2019-11-23 20:27:39 +08:00
|
|
|
#include "platform/android/jni/JniHelper.h"
|
2023-09-02 19:56:50 +08:00
|
|
|
#include "platform/GL.h"
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <android/log.h>
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
#define DEFAULT_MARGIN_ANDROID 30.0f
|
|
|
|
#define WIDE_SCREEN_ASPECT_RATIO_ANDROID 2.0f
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2023-09-01 16:31:14 +08:00
|
|
|
|
2023-09-02 19:56:33 +08:00
|
|
|
NS_AX_BEGIN
|
2023-09-02 19:56:50 +08:00
|
|
|
void GLViewImpl::loadGLES2()
|
|
|
|
{
|
|
|
|
auto glesVer = gladLoaderLoadGLES2();
|
|
|
|
if (glesVer)
|
2024-03-07 08:47:00 +08:00
|
|
|
AXLOGI("Load GLES success, version: {}", glesVer);
|
2023-09-02 19:56:50 +08:00
|
|
|
else
|
|
|
|
throw std::runtime_error("Load GLES fail");
|
|
|
|
}
|
2023-09-02 19:56:33 +08:00
|
|
|
|
2023-03-25 08:37:51 +08:00
|
|
|
GLViewImpl* GLViewImpl::createWithRect(std::string_view viewName, const Rect& rect, float frameZoomFactor, bool resizable)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
auto ret = new GLViewImpl;
|
2023-03-25 08:37:51 +08:00
|
|
|
if (ret && ret->initWithRect(viewName, rect, frameZoomFactor, resizable))
|
2021-12-25 10:04:45 +08:00
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
ret->autorelease();
|
|
|
|
return ret;
|
|
|
|
}
|
2022-07-16 10:43:05 +08:00
|
|
|
AX_SAFE_DELETE(ret);
|
2019-11-23 20:27:39 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
GLViewImpl* GLViewImpl::create(std::string_view viewName)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
auto ret = new GLViewImpl;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (ret && ret->initWithFullScreen(viewName))
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
ret->autorelease();
|
|
|
|
return ret;
|
|
|
|
}
|
2022-07-16 10:43:05 +08:00
|
|
|
AX_SAFE_DELETE(ret);
|
2019-11-23 20:27:39 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
GLViewImpl* GLViewImpl::createWithFullScreen(std::string_view viewName)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
auto ret = new GLViewImpl();
|
2021-12-25 10:04:45 +08:00
|
|
|
if (ret && ret->initWithFullScreen(viewName))
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
ret->autorelease();
|
|
|
|
return ret;
|
|
|
|
}
|
2022-07-16 10:43:05 +08:00
|
|
|
AX_SAFE_DELETE(ret);
|
2019-11-23 20:27:39 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLViewImpl::GLViewImpl()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
GLViewImpl::~GLViewImpl() {}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2023-03-25 08:37:51 +08:00
|
|
|
bool GLViewImpl::initWithRect(std::string_view /*viewName*/, const Rect& /*rect*/, float /*frameZoomFactor*/, bool /*resizable*/)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-12-31 12:12:40 +08:00
|
|
|
bool GLViewImpl::initWithFullScreen(std::string_view viewName)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GLViewImpl::isOpenGLReady()
|
|
|
|
{
|
|
|
|
return (_screenSize.width != 0 && _screenSize.height != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GLViewImpl::end()
|
|
|
|
{
|
2023-05-25 15:45:00 +08:00
|
|
|
JniHelper::callStaticVoidMethod("org.axmol.lib.AxmolEngine", "onExit");
|
|
|
|
release();
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
void GLViewImpl::swapBuffers() {}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
void GLViewImpl::setIMEKeyboardState(bool bOpen)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
if (bOpen)
|
|
|
|
{
|
2022-10-01 16:24:52 +08:00
|
|
|
JniHelper::callStaticVoidMethod("org.axmol.lib.AxmolGLSurfaceView", "openIMEKeyboard");
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-10-01 16:24:52 +08:00
|
|
|
JniHelper::callStaticVoidMethod("org.axmol.lib.AxmolGLSurfaceView", "closeIMEKeyboard");
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
Rect GLViewImpl::getSafeAreaRect() const
|
|
|
|
{
|
|
|
|
Rect safeAreaRect = GLView::getSafeAreaRect();
|
2019-11-23 20:27:39 +08:00
|
|
|
float deviceAspectRatio = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
if (safeAreaRect.size.height > safeAreaRect.size.width)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
deviceAspectRatio = safeAreaRect.size.height / safeAreaRect.size.width;
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
deviceAspectRatio = safeAreaRect.size.width / safeAreaRect.size.height;
|
|
|
|
}
|
|
|
|
|
|
|
|
float marginX = DEFAULT_MARGIN_ANDROID / _scaleX;
|
|
|
|
float marginY = DEFAULT_MARGIN_ANDROID / _scaleY;
|
|
|
|
|
2022-10-01 16:24:52 +08:00
|
|
|
bool isScreenRound = JniHelper::callStaticBooleanMethod("org/axmol/lib/AxmolEngine", "isScreenRound");
|
|
|
|
bool hasSoftKeys = JniHelper::callStaticBooleanMethod("org/axmol/lib/AxmolEngine", "hasSoftKeys");
|
|
|
|
bool isCutoutEnabled = JniHelper::callStaticBooleanMethod("org/axmol/lib/AxmolEngine", "isCutoutEnabled");
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2024-05-28 20:06:15 +08:00
|
|
|
float insetTop = 0.0f;
|
|
|
|
float insetBottom = 0.0f;
|
|
|
|
float insetLeft = 0.0f;
|
|
|
|
float insetRight = 0.0f;
|
|
|
|
|
|
|
|
static int* cornerRadii =
|
|
|
|
JniHelper::callStaticIntArrayMethod("org/axmol/lib/AxmolEngine", "getDeviceCornerRadii");
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (isScreenRound)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
// edge screen (ex. Samsung Galaxy s7, s9, s9+, Note 9, Nokia 8 Sirocco, Sony Xperia XZ3, Oppo Find X...)
|
2021-12-25 10:04:45 +08:00
|
|
|
if (safeAreaRect.size.width < safeAreaRect.size.height)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
safeAreaRect.origin.y += marginY * 2.f;
|
|
|
|
safeAreaRect.size.height -= (marginY * 2.f);
|
|
|
|
|
|
|
|
safeAreaRect.origin.x += marginX;
|
|
|
|
safeAreaRect.size.width -= (marginX * 2.f);
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
safeAreaRect.origin.y += marginY;
|
|
|
|
safeAreaRect.size.height -= (marginY * 2.f);
|
|
|
|
|
|
|
|
// landscape: no changes with X-coords
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
2024-05-28 20:06:15 +08:00
|
|
|
else if (deviceAspectRatio >= WIDE_SCREEN_ASPECT_RATIO_ANDROID || cornerRadii != nullptr)
|
2021-12-25 10:04:45 +08:00
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
// almost all devices on the market have round corners if
|
|
|
|
// deviceAspectRatio more than 2 (@see "android.max_aspect" parameter in AndroidManifest.xml)
|
|
|
|
|
2024-05-28 20:06:15 +08:00
|
|
|
// cornerRadii is only available in API31+ (Android 12+)
|
|
|
|
if (cornerRadii != nullptr)
|
2021-12-25 10:04:45 +08:00
|
|
|
{
|
2024-05-28 20:06:15 +08:00
|
|
|
float radiiBottom = cornerRadii[0] / _scaleY;
|
|
|
|
float radiiLeft = cornerRadii[1] / _scaleX;
|
|
|
|
float radiiRight = cornerRadii[2] / _scaleX;
|
|
|
|
float radiiTop = cornerRadii[3] / _scaleY;
|
|
|
|
|
|
|
|
if (safeAreaRect.size.width < safeAreaRect.size.height)
|
|
|
|
{
|
|
|
|
if (hasSoftKeys)
|
|
|
|
{
|
|
|
|
safeAreaRect.origin.y += marginY;
|
|
|
|
safeAreaRect.size.height -= (marginY * 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// portrait
|
|
|
|
insetTop = radiiTop;
|
|
|
|
insetBottom = radiiBottom;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// landscape
|
|
|
|
insetLeft = radiiLeft;
|
|
|
|
insetRight = radiiRight;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-05-28 20:06:15 +08:00
|
|
|
float bottomMarginIfPortrait = 0;
|
|
|
|
if (hasSoftKeys)
|
|
|
|
{
|
|
|
|
bottomMarginIfPortrait = marginY * 2.f;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (safeAreaRect.size.width < safeAreaRect.size.height)
|
|
|
|
{
|
|
|
|
// portrait: double margin space if device has soft menu
|
|
|
|
safeAreaRect.origin.y += bottomMarginIfPortrait;
|
|
|
|
safeAreaRect.size.height -= (bottomMarginIfPortrait + marginY);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// landscape: ignore double margin at the bottom in any cases
|
|
|
|
// prepare single margin for round corners
|
|
|
|
safeAreaRect.origin.y += marginY;
|
|
|
|
safeAreaRect.size.height -= (marginY * 2.f);
|
|
|
|
}
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (hasSoftKeys && (safeAreaRect.size.width < safeAreaRect.size.height))
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
// portrait: preserve only for soft system menu
|
|
|
|
safeAreaRect.origin.y += marginY * 2.f;
|
|
|
|
safeAreaRect.size.height -= (marginY * 2.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (isCutoutEnabled)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
// screen with enabled cutout area (ex. Google Pixel 3 XL, Huawei P20, Asus ZenFone 5, etc)
|
2021-12-25 10:04:45 +08:00
|
|
|
static int* safeInsets =
|
2022-10-01 16:24:52 +08:00
|
|
|
JniHelper::callStaticIntArrayMethod("org/axmol/lib/AxmolEngine", "getSafeInsets");
|
2021-12-25 10:04:45 +08:00
|
|
|
if (safeInsets != nullptr)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
float safeInsetBottom = safeInsets[0] / _scaleY;
|
2021-12-25 10:04:45 +08:00
|
|
|
float safeInsetLeft = safeInsets[1] / _scaleX;
|
|
|
|
float safeInsetRight = safeInsets[2] / _scaleX;
|
|
|
|
float safeInsetTop = safeInsets[3] / _scaleY;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
// fit safe area rect with safe insets
|
2024-05-28 20:06:15 +08:00
|
|
|
auto maxInsetBottom = std::max(safeInsetBottom, insetBottom);
|
|
|
|
safeAreaRect.origin.y += maxInsetBottom;
|
|
|
|
safeAreaRect.size.height -= maxInsetBottom;
|
|
|
|
|
|
|
|
auto maxInsetLeft = std::max(safeInsetLeft, insetLeft);
|
|
|
|
safeAreaRect.origin.x += maxInsetLeft;
|
|
|
|
safeAreaRect.size.width -= maxInsetLeft;
|
|
|
|
|
|
|
|
safeAreaRect.size.width -= std::max(safeInsetRight, insetRight);
|
|
|
|
safeAreaRect.size.height -= std::max(safeInsetTop, insetTop);
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return safeAreaRect;
|
|
|
|
}
|
|
|
|
|
2023-05-31 20:42:39 +08:00
|
|
|
void GLViewImpl::queueOperation(void (*op)(void*), void* param)
|
|
|
|
{
|
|
|
|
JniHelper::callStaticVoidMethod("org.axmol.lib.AxmolEngine", "queueOperation", (jlong)(uintptr_t)op,
|
|
|
|
(jlong)(uintptr_t)param);
|
|
|
|
}
|
|
|
|
|
2022-07-11 17:50:21 +08:00
|
|
|
NS_AX_END
|