diff --git a/android/jni/CMakeLists.txt b/android/jni/CMakeLists.txt index da2a537112..0f63b4a5d4 100644 --- a/android/jni/CMakeLists.txt +++ b/android/jni/CMakeLists.txt @@ -32,7 +32,7 @@ set(SRC app/organicmaps/isolines/IsolinesManager.cpp app/organicmaps/LocationHelper.cpp app/organicmaps/LocationState.cpp - app/organicmaps/MapFragment.cpp + app/organicmaps/Map.cpp app/organicmaps/MapManager.cpp app/organicmaps/MwmApplication.cpp app/organicmaps/routing/RoutingOptions.cpp diff --git a/android/jni/app/organicmaps/Framework.cpp b/android/jni/app/organicmaps/Framework.cpp index 5ba1119c28..ad17613965 100644 --- a/android/jni/app/organicmaps/Framework.cpp +++ b/android/jni/app/organicmaps/Framework.cpp @@ -471,6 +471,16 @@ void Framework::ShowNode(CountryId const & idx, bool zoomToDownloadButton) } } +void Framework::Scale(double factor, m2::PointD const & pxPoint, bool isAnim) +{ + m_work.Scale(factor, pxPoint, isAnim); +} + +void Framework::Move(double factorX, double factorY, bool isAnim) +{ + m_work.Move(factorX, factorY, isAnim); +} + void Framework::Touch(int action, Finger const & f1, Finger const & f2, uint8_t maskedPointer) { MultiTouchAction eventType = static_cast(action); diff --git a/android/jni/app/organicmaps/Framework.hpp b/android/jni/app/organicmaps/Framework.hpp index 22ccdc41d6..9cc3d5e265 100644 --- a/android/jni/app/organicmaps/Framework.hpp +++ b/android/jni/app/organicmaps/Framework.hpp @@ -127,6 +127,10 @@ namespace android float m_x, m_y; }; + void Scale(double factor, m2::PointD const & pxPoint, bool isAnim); + + void Move(double factorX, double factorY, bool isAnim); + void Touch(int action, Finger const & f1, Finger const & f2, uint8_t maskedPointer); bool Search(search::EverywhereSearchParams const & params); diff --git a/android/jni/app/organicmaps/MapFragment.cpp b/android/jni/app/organicmaps/Map.cpp similarity index 51% rename from android/jni/app/organicmaps/MapFragment.cpp rename to android/jni/app/organicmaps/Map.cpp index 0422dae14a..616b5dbb5b 100644 --- a/android/jni/app/organicmaps/MapFragment.cpp +++ b/android/jni/app/organicmaps/Map.cpp @@ -22,138 +22,32 @@ void OnRenderingInitializationFinished(std::shared_ptr const & listener extern "C" { -using namespace storage; - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeCompassUpdated(JNIEnv * env, jclass clazz, jdouble north, jboolean forceRedraw) -{ - location::CompassInfo info; - info.m_bearing = north; - - g_framework->OnCompassUpdated(info, forceRedraw); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeStorageConnected(JNIEnv * env, jclass clazz) -{ - android::Platform::Instance().OnExternalStorageStatusChanged(true); - g_framework->AddLocalMaps(); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeStorageDisconnected(JNIEnv * env, jclass clazz) -{ - android::Platform::Instance().OnExternalStorageStatusChanged(false); - g_framework->RemoveLocalMaps(); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeScalePlus(JNIEnv * env, jclass clazz) -{ - g_framework->Scale(::Framework::SCALE_MAG); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeScaleMinus(JNIEnv * env, jclass clazz) -{ - g_framework->Scale(::Framework::SCALE_MIN); -} - JNIEXPORT jboolean JNICALL -Java_app_organicmaps_MapFragment_nativeShowMapForUrl(JNIEnv * env, jclass clazz, jstring url) -{ - return g_framework->ShowMapForURL(jni::ToNativeString(env, url)); -} - -JNIEXPORT jboolean JNICALL -Java_app_organicmaps_MapFragment_nativeCreateEngine(JNIEnv * env, jclass clazz, - jobject surface, jint density, - jboolean firstLaunch, - jboolean isLaunchByDeepLink, - jint appVersionCode) +Java_app_organicmaps_Map_nativeCreateEngine(JNIEnv * env, jclass, + jobject surface, jint density, + jboolean firstLaunch, + jboolean isLaunchByDeepLink, + jint appVersionCode) { return g_framework->CreateDrapeEngine(env, surface, density, firstLaunch, isLaunchByDeepLink, base::asserted_cast(appVersionCode)); } JNIEXPORT jboolean JNICALL -Java_app_organicmaps_MapFragment_nativeIsEngineCreated(JNIEnv * env, jclass clazz) +Java_app_organicmaps_Map_nativeIsEngineCreated(JNIEnv *, jclass) { return g_framework->IsDrapeEngineCreated(); } JNIEXPORT jboolean JNICALL -Java_app_organicmaps_MapFragment_nativeDestroySurfaceOnDetach(JNIEnv * env, jclass clazz) +Java_app_organicmaps_Map_nativeShowMapForUrl(JNIEnv * env, jclass, jstring url) { - return g_framework->DestroySurfaceOnDetach(); -} - -JNIEXPORT jboolean JNICALL -Java_app_organicmaps_MapFragment_nativeAttachSurface(JNIEnv * env, jclass clazz, - jobject surface) -{ - return g_framework->AttachSurface(env, surface); + return g_framework->ShowMapForURL(jni::ToNativeString(env, url)); } JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeDetachSurface(JNIEnv * env, jclass clazz, - jboolean destroySurface) -{ - g_framework->DetachSurface(destroySurface); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativePauseSurfaceRendering(JNIEnv *, jclass) -{ - g_framework->PauseSurfaceRendering(); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeResumeSurfaceRendering(JNIEnv *, jclass) -{ - g_framework->ResumeSurfaceRendering(); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeSurfaceChanged(JNIEnv * env, jclass, jobject surface, - jint w, jint h) -{ - g_framework->Resize(env, surface, w, h); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeOnTouch(JNIEnv * env, jclass clazz, jint action, - jint id1, jfloat x1, jfloat y1, - jint id2, jfloat x2, jfloat y2, - jint maskedPointer) -{ - g_framework->Touch(action, - android::Framework::Finger(id1, x1, y1), - android::Framework::Finger(id2, x2, y2), maskedPointer); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeSetupWidget( - JNIEnv * env, jclass clazz, jint widget, jfloat x, jfloat y, jint anchor) -{ - g_framework->SetupWidget(static_cast(widget), x, y, static_cast(anchor)); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeApplyWidgets(JNIEnv * env, jclass clazz) -{ - g_framework->ApplyWidgets(); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeCleanWidgets(JNIEnv * env, jclass clazz) -{ - g_framework->CleanWidgets(); -} - -JNIEXPORT void JNICALL -Java_app_organicmaps_MapFragment_nativeSetRenderingInitializationFinishedListener( - JNIEnv * env, jclass clazz, jobject listener) +Java_app_organicmaps_Map_nativeSetRenderingInitializationFinishedListener( + JNIEnv *, jclass, jobject listener) { if (listener) { @@ -165,4 +59,119 @@ Java_app_organicmaps_MapFragment_nativeSetRenderingInitializationFinishedListene g_framework->NativeFramework()->SetGraphicsContextInitializationHandler(nullptr); } } + +JNIEXPORT jboolean JNICALL +Java_app_organicmaps_Map_nativeAttachSurface(JNIEnv * env, jclass, jobject surface) +{ + return g_framework->AttachSurface(env, surface); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeDetachSurface(JNIEnv *, jclass, jboolean destroySurface) +{ + g_framework->DetachSurface(destroySurface); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeSurfaceChanged(JNIEnv * env, jclass, jobject surface, jint w, jint h) +{ + g_framework->Resize(env, surface, w, h); +} + +JNIEXPORT jboolean JNICALL +Java_app_organicmaps_Map_nativeDestroySurfaceOnDetach(JNIEnv *, jclass) +{ + return g_framework->DestroySurfaceOnDetach(); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativePauseSurfaceRendering(JNIEnv *, jclass) +{ + g_framework->PauseSurfaceRendering(); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeResumeSurfaceRendering(JNIEnv *, jclass) +{ + g_framework->ResumeSurfaceRendering(); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeApplyWidgets(JNIEnv *, jclass) +{ + g_framework->ApplyWidgets(); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeCleanWidgets(JNIEnv *, jclass) +{ + g_framework->CleanWidgets(); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeSetupWidget( + JNIEnv *, jclass, jint widget, jfloat x, jfloat y, jint anchor) +{ + g_framework->SetupWidget(static_cast(widget), x, y, static_cast(anchor)); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeCompassUpdated(JNIEnv *, jclass, jdouble north, jboolean forceRedraw) +{ + location::CompassInfo info; + info.m_bearing = north; + + g_framework->OnCompassUpdated(info, forceRedraw); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeMove( + JNIEnv *, jclass, jdouble factorX, jdouble factorY, jboolean isAnim) +{ + g_framework->Move(factorX, factorY, isAnim); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeScalePlus(JNIEnv *, jclass) +{ + g_framework->Scale(::Framework::SCALE_MAG); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeScaleMinus(JNIEnv *, jclass) +{ + g_framework->Scale(::Framework::SCALE_MIN); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeScale( + JNIEnv *, jclass, jdouble factor, jdouble focusX, jdouble focusY, jboolean isAnim) +{ + g_framework->Scale(factor, {focusX, focusY}, isAnim); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeOnTouch(JNIEnv *, jclass, jint action, + jint id1, jfloat x1, jfloat y1, + jint id2, jfloat x2, jfloat y2, + jint maskedPointer) +{ + g_framework->Touch(action, + android::Framework::Finger(id1, x1, y1), + android::Framework::Finger(id2, x2, y2), maskedPointer); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeStorageConnected(JNIEnv *, jclass) +{ + android::Platform::Instance().OnExternalStorageStatusChanged(true); + g_framework->AddLocalMaps(); +} + +JNIEXPORT void JNICALL +Java_app_organicmaps_Map_nativeStorageDisconnected(JNIEnv *, jclass) +{ + android::Platform::Instance().OnExternalStorageStatusChanged(false); + g_framework->RemoveLocalMaps(); +} } // extern "C" diff --git a/android/src/app/organicmaps/Map.java b/android/src/app/organicmaps/Map.java new file mode 100644 index 0000000000..df547cef44 --- /dev/null +++ b/android/src/app/organicmaps/Map.java @@ -0,0 +1,365 @@ +package app.organicmaps; + +import android.content.Context; +import android.graphics.Rect; +import android.view.MotionEvent; +import android.view.Surface; + +import androidx.annotation.Nullable; + +import app.organicmaps.location.LocationHelper; +import app.organicmaps.util.Config; +import app.organicmaps.util.UiUtils; +import app.organicmaps.util.concurrency.UiThread; +import app.organicmaps.util.log.Logger; + +public final class Map +{ + public interface CallbackUnsupported + { + void report(); + } + + public static final String ARG_LAUNCH_BY_DEEP_LINK = "launch_by_deep_link"; + private static final String TAG = Map.class.getSimpleName(); + + // Should correspond to android::MultiTouchAction from Framework.cpp + public static final int NATIVE_ACTION_UP = 0x01; + public static final int NATIVE_ACTION_DOWN = 0x02; + public static final int NATIVE_ACTION_MOVE = 0x03; + public static final int NATIVE_ACTION_CANCEL = 0x04; + + // Should correspond to gui::EWidget from skin.hpp + public static final int WIDGET_RULER = 0x01; + public static final int WIDGET_COMPASS = 0x02; + public static final int WIDGET_COPYRIGHT = 0x04; + public static final int WIDGET_SCALE_FPS_LABEL = 0x08; + + // Should correspond to dp::Anchor from drape_global.hpp + public static final int ANCHOR_CENTER = 0x00; + public static final int ANCHOR_LEFT = 0x01; + public static final int ANCHOR_RIGHT = (ANCHOR_LEFT << 1); + public static final int ANCHOR_TOP = (ANCHOR_RIGHT << 1); + public static final int ANCHOR_BOTTOM = (ANCHOR_TOP << 1); + public static final int ANCHOR_LEFT_TOP = (ANCHOR_LEFT | ANCHOR_TOP); + public static final int ANCHOR_RIGHT_TOP = (ANCHOR_RIGHT | ANCHOR_TOP); + public static final int ANCHOR_LEFT_BOTTOM = (ANCHOR_LEFT | ANCHOR_BOTTOM); + public static final int ANCHOR_RIGHT_BOTTOM = (ANCHOR_RIGHT | ANCHOR_BOTTOM); + + // Should correspond to df::TouchEvent::INVALID_MASKED_POINTER from user_event_stream.cpp + public static final int INVALID_POINTER_MASK = 0xFF; + public static final int INVALID_TOUCH_ID = -1; + + private int mCurrentCompassOffsetX; + private int mCurrentCompassOffsetY; + private int mBottomWidgetOffsetX; + private int mBottomWidgetOffsetY; + + private int mHeight; + private int mWidth; + private boolean mRequireResize; + private boolean mSurfaceCreated; + private boolean mSurfaceAttached; + private boolean mLaunchByDeepLink; + @Nullable + private String mUiThemeOnPause; + @Nullable + private MapRenderingListener mMapRenderingListener; + @Nullable + private CallbackUnsupported mCallbackUnsupported; + + public Map() + { + onCreate(false); + } + + /** + * Moves the map compass using the given offsets. + * + * @param context Context. + * @param offsetX Pixel offset from the top. -1 to keep the previous value. + * @param offsetY Pixel offset from the right. -1 to keep the previous value. + * @param forceRedraw True to force the compass to redraw + */ + public void setupCompass(final Context context, int offsetX, int offsetY, boolean forceRedraw) + { + final int x = offsetX < 0 ? mCurrentCompassOffsetX : offsetX; + final int y = offsetY < 0 ? mCurrentCompassOffsetY : offsetY; + final int navPadding = UiUtils.dimen(context, R.dimen.nav_frame_padding); + final int marginX = UiUtils.dimen(context, R.dimen.margin_compass) + navPadding; + final int marginY = UiUtils.dimen(context, R.dimen.margin_compass_top) + navPadding; + nativeSetupWidget(WIDGET_COMPASS, mWidth - x - marginX, y + marginY, ANCHOR_CENTER); + if (forceRedraw && mSurfaceCreated) + nativeApplyWidgets(); + mCurrentCompassOffsetX = x; + mCurrentCompassOffsetY = y; + } + + public static void onCompassUpdated(double north, boolean forceRedraw) + { + nativeCompassUpdated(north, forceRedraw); + } + + /** + * Moves the ruler and copyright using the given offsets. + * + * @param context Context. + * @param offsetX Pixel offset from the left. -1 to keep the previous value. + * @param offsetY Pixel offset from the bottom. -1 to keep the previous value. + */ + public void setupBottomWidgetsOffset(final Context context, int offsetX, int offsetY) + { + final int x = offsetX < 0 ? mBottomWidgetOffsetX : offsetX; + final int y = offsetY < 0 ? mBottomWidgetOffsetY : offsetY; + setupRuler(context, x, y); + setupAttribution(context, x, y); + mBottomWidgetOffsetX = x; + mBottomWidgetOffsetY = y; + } + + public void onSurfaceCreated(final Context context, final Surface surface, Rect surfaceFrame, int surfaceDpi) + { + if (nativeIsEngineCreated()) + nativeDetachSurface(true); + + if (isThemeChangingProcess(context)) + { + Logger.d(TAG, "Theme changing process, skip 'onSurfaceCreated' callback"); + return; + } + + Logger.d(TAG, "mSurfaceCreated = " + mSurfaceCreated); + if (nativeIsEngineCreated()) + { + if (!nativeAttachSurface(surface)) + { + if (mCallbackUnsupported != null) + mCallbackUnsupported.report(); + return; + } + mSurfaceCreated = true; + mSurfaceAttached = true; + mRequireResize = true; + nativeResumeSurfaceRendering(); + return; + } + + mRequireResize = false; + setupWidgets(context, surfaceFrame.width(), surfaceFrame.height()); + + final boolean firstStart = LocationHelper.INSTANCE.isInFirstRun(); + if (!nativeCreateEngine(surface, surfaceDpi, firstStart, mLaunchByDeepLink, BuildConfig.VERSION_CODE)) + { + if (mCallbackUnsupported != null) + mCallbackUnsupported.report(); + return; + } + + if (firstStart) + UiThread.runLater(LocationHelper.INSTANCE::onExitFromFirstRun); + + mSurfaceCreated = true; + mSurfaceAttached = true; + nativeResumeSurfaceRendering(); + if (mMapRenderingListener != null) + mMapRenderingListener.onRenderingCreated(); + } + + public void onSurfaceChanged(final Context context, final Surface surface, Rect surfaceFrame, boolean isSurfaceCreating) + { + if (isThemeChangingProcess(context)) + { + Logger.d(TAG, "Theme changing process, skip 'onSurfaceChanged' callback"); + return; + } + + Logger.d(TAG, "mSurfaceCreated = " + mSurfaceCreated); + if (!mSurfaceCreated || (!mRequireResize && isSurfaceCreating)) + return; + + nativeSurfaceChanged(surface, surfaceFrame.width(), surfaceFrame.height()); + + mRequireResize = false; + setupWidgets(context, surfaceFrame.width(), surfaceFrame.height()); + nativeApplyWidgets(); + if (mMapRenderingListener != null) + mMapRenderingListener.onRenderingRestored(); + } + + public void onSurfaceDestroyed(boolean activityIsChangingConfigurations, boolean isAdded) + { + Logger.d(TAG, "mSurfaceCreated = " + mSurfaceCreated + ", mSurfaceAttached = " + mSurfaceAttached + ", isAdded = " + isAdded); + if (!mSurfaceCreated || !mSurfaceAttached || !isAdded) + return; + + nativeDetachSurface(!activityIsChangingConfigurations); + mSurfaceCreated = !nativeDestroySurfaceOnDetach(); + mSurfaceAttached = false; + } + + public void setMapRenderingListener(MapRenderingListener mapRenderingListener) + { + mMapRenderingListener = mapRenderingListener; + } + + public void setCallbackUnsupported(CallbackUnsupported callback) + { + mCallbackUnsupported = callback; + } + + public void onCreate(boolean launchByDeeplink) + { + mLaunchByDeepLink = launchByDeeplink; + mCurrentCompassOffsetX = 0; + mCurrentCompassOffsetY = 0; + mBottomWidgetOffsetX = 0; + mBottomWidgetOffsetY = 0; + } + + public void onStart() + { + nativeSetRenderingInitializationFinishedListener(mMapRenderingListener); + } + + public void onStop() + { + nativeSetRenderingInitializationFinishedListener(null); + } + + public void onPause(final Context context) + { + mUiThemeOnPause = Config.getCurrentUiTheme(context); + + // Pause/Resume can be called without surface creation/destroy. + if (mSurfaceAttached) + nativePauseSurfaceRendering(); + } + + public void onResume() + { + // Pause/Resume can be called without surface creation/destroy. + if (mSurfaceAttached) + nativeResumeSurfaceRendering(); + } + + boolean isContextCreated() + { + return mSurfaceCreated; + } + + public void onScroll(float distanceX, float distanceY) + { + Map.nativeMove(-distanceX / ((float) mWidth), distanceY / ((float) mHeight), false); + } + + public static void zoomIn() + { + nativeScalePlus(); + } + + public static void zoomOut() + { + nativeScaleMinus(); + } + + public static void onScale(double factor, double focusX, double focusY, boolean isAnim) + { + nativeScale(factor, focusX, focusY, isAnim); + } + + public static void onTouch(int actionType, MotionEvent event, int pointerIndex) + { + if (event.getPointerCount() == 1) + { + nativeOnTouch(actionType, event.getPointerId(0), event.getX(), event.getY(), Map.INVALID_TOUCH_ID, 0, 0, 0); + } + else + { + nativeOnTouch(actionType, + event.getPointerId(0), event.getX(0), event.getY(0), + event.getPointerId(1), event.getX(1), event.getY(1), pointerIndex); + } + } + + public static void onTouch(float x, float y) + { + nativeOnTouch(Map.NATIVE_ACTION_UP, 0, x, y, Map.INVALID_TOUCH_ID, 0, 0, 0); + } + + public static boolean isEngineCreated() + { + return nativeIsEngineCreated(); + } + + public static boolean showMapForUrl(String url) + { + return nativeShowMapForUrl(url); + } + + private void setupWidgets(final Context context, int width, int height) + { + mHeight = height; + mWidth = width; + + nativeCleanWidgets(); + setupBottomWidgetsOffset(context, mBottomWidgetOffsetX, mBottomWidgetOffsetY); + nativeSetupWidget(WIDGET_SCALE_FPS_LABEL, UiUtils.dimen(context, R.dimen.margin_base), UiUtils.dimen(context, R.dimen.margin_base), ANCHOR_LEFT_TOP); + setupCompass(context, mCurrentCompassOffsetX, mCurrentCompassOffsetY, false); + } + + private void setupRuler(final Context context, int offsetX, int offsetY) + { + nativeSetupWidget(WIDGET_RULER, + UiUtils.dimen(context, R.dimen.margin_ruler) + offsetX, + mHeight - UiUtils.dimen(context, R.dimen.margin_ruler) - offsetY, + ANCHOR_LEFT_BOTTOM); + if (mSurfaceCreated) + nativeApplyWidgets(); + } + + private void setupAttribution(final Context context, int offsetX, int offsetY) + { + nativeSetupWidget(WIDGET_COPYRIGHT, + UiUtils.dimen(context, R.dimen.margin_ruler) + offsetX, + mHeight - UiUtils.dimen(context, R.dimen.margin_ruler) - offsetY, + ANCHOR_LEFT_BOTTOM); + if (mSurfaceCreated) + nativeApplyWidgets(); + } + + private boolean isThemeChangingProcess(final Context context) + { + return mUiThemeOnPause != null && !mUiThemeOnPause.equals(Config.getCurrentUiTheme(context)); + } + + // Engine + private static native boolean nativeCreateEngine(Surface surface, int density, + boolean firstLaunch, + boolean isLaunchByDeepLink, + int appVersionCode); + private static native boolean nativeIsEngineCreated(); + private static native void nativeSetRenderingInitializationFinishedListener( + @Nullable MapRenderingListener listener); + private static native boolean nativeShowMapForUrl(String url); + + // Surface + private static native boolean nativeAttachSurface(Surface surface); + private static native void nativeDetachSurface(boolean destroySurface); + private static native void nativeSurfaceChanged(Surface surface, int w, int h); + private static native boolean nativeDestroySurfaceOnDetach(); + private static native void nativePauseSurfaceRendering(); + private static native void nativeResumeSurfaceRendering(); + + // Widgets + private static native void nativeApplyWidgets(); + private static native void nativeCleanWidgets(); + private static native void nativeSetupWidget(int widget, float x, float y, int anchor); + private static native void nativeCompassUpdated(double north, boolean forceRedraw); + + // Events + private static native void nativeMove(double factorX, double factorY, boolean isAnim); + private static native void nativeScalePlus(); + private static native void nativeScaleMinus(); + private static native void nativeScale(double factor, double focusX, double focusY, boolean isAnim); + private static native void nativeOnTouch(int actionType, int id1, float x1, float y1, int id2, float x2, float y2, int maskedPointer); +} diff --git a/android/src/app/organicmaps/MapFragment.java b/android/src/app/organicmaps/MapFragment.java index e060fdda93..326d8cfe59 100644 --- a/android/src/app/organicmaps/MapFragment.java +++ b/android/src/app/organicmaps/MapFragment.java @@ -1,251 +1,65 @@ package app.organicmaps; import android.content.Context; -import android.graphics.Rect; +import android.os.Build; import android.os.Bundle; import android.util.DisplayMetrics; import android.view.LayoutInflater; import android.view.MotionEvent; -import android.view.Surface; import android.view.SurfaceHolder; import android.view.SurfaceView; import android.view.View; import android.view.ViewGroup; -import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.appcompat.app.AlertDialog; +import androidx.core.content.res.ConfigurationHelper; import app.organicmaps.base.BaseMwmFragment; -import app.organicmaps.location.LocationHelper; -import app.organicmaps.util.Config; -import app.organicmaps.util.UiUtils; -import app.organicmaps.util.concurrency.UiThread; import app.organicmaps.util.log.Logger; -public class MapFragment extends BaseMwmFragment - implements View.OnTouchListener, - SurfaceHolder.Callback +public class MapFragment extends BaseMwmFragment implements View.OnTouchListener, SurfaceHolder.Callback { - public static final String ARG_LAUNCH_BY_DEEP_LINK = "launch_by_deep_link"; private static final String TAG = MapFragment.class.getSimpleName(); + private final Map mMap = new Map(); - // Should correspond to android::MultiTouchAction from Framework.cpp - private static final int NATIVE_ACTION_UP = 0x01; - private static final int NATIVE_ACTION_DOWN = 0x02; - private static final int NATIVE_ACTION_MOVE = 0x03; - private static final int NATIVE_ACTION_CANCEL = 0x04; - - // Should correspond to gui::EWidget from skin.hpp - private static final int WIDGET_RULER = 0x01; - private static final int WIDGET_COMPASS = 0x02; - private static final int WIDGET_COPYRIGHT = 0x04; - private static final int WIDGET_SCALE_FPS_LABEL = 0x08; - - // Should correspond to dp::Anchor from drape_global.hpp - private static final int ANCHOR_CENTER = 0x00; - private static final int ANCHOR_LEFT = 0x01; - private static final int ANCHOR_RIGHT = (ANCHOR_LEFT << 1); - private static final int ANCHOR_TOP = (ANCHOR_RIGHT << 1); - private static final int ANCHOR_BOTTOM = (ANCHOR_TOP << 1); - private static final int ANCHOR_LEFT_TOP = (ANCHOR_LEFT | ANCHOR_TOP); - private static final int ANCHOR_RIGHT_TOP = (ANCHOR_RIGHT | ANCHOR_TOP); - private static final int ANCHOR_LEFT_BOTTOM = (ANCHOR_LEFT | ANCHOR_BOTTOM); - private static final int ANCHOR_RIGHT_BOTTOM = (ANCHOR_RIGHT | ANCHOR_BOTTOM); - - // Should correspond to df::TouchEvent::INVALID_MASKED_POINTER from user_event_stream.cpp - private static final int INVALID_POINTER_MASK = 0xFF; - private static final int INVALID_TOUCH_ID = -1; - - private int mCurrentCompassOffsetX; - private int mCurrentCompassOffsetY; - private int mBottomWidgetOffsetX; - private int mBottomWidgetOffsetY; - - private int mHeight; - private int mWidth; - private boolean mRequireResize; - private boolean mSurfaceCreated; - private boolean mSurfaceAttached; - private boolean mLaunchByDeepLink; - @Nullable - private String mUiThemeOnPause; - @SuppressWarnings("NullableProblems") - @NonNull - private SurfaceView mSurfaceView; - @Nullable - private MapRenderingListener mMapRenderingListener; - - private void setupWidgets(int width, int height) + public void adjustCompass(int offsetX, int offsetY) { - mHeight = height; - mWidth = width; - Context context = requireContext(); - - nativeCleanWidgets(); - setupBottomWidgetsOffset(mBottomWidgetOffsetY, mBottomWidgetOffsetX); - - nativeSetupWidget(WIDGET_SCALE_FPS_LABEL, - UiUtils.dimen(context, R.dimen.margin_base), - UiUtils.dimen(context, R.dimen.margin_base), - ANCHOR_LEFT_TOP); - - setupCompass(mCurrentCompassOffsetY, mCurrentCompassOffsetX, false); + mMap.setupCompass(requireContext(), offsetX, offsetY, true); } - /** - * Moves the map compass using the given offsets. - * - * @param offsetY Pixel offset from the top. -1 to keep the previous value. - * @param offsetX Pixel offset from the right. -1 to keep the previous value. - * @param forceRedraw True to force the compass to redraw - */ - void setupCompass(int offsetY, int offsetX, boolean forceRedraw) + public void adjustBottomWidgets(int offsetX, int offsetY) { - Context context = requireContext(); - int x = offsetX < 0 ? mCurrentCompassOffsetX : offsetX; - int y = offsetY < 0 ? mCurrentCompassOffsetY : offsetY; - int navPadding = UiUtils.dimen(context, R.dimen.nav_frame_padding); - int marginX = UiUtils.dimen(context, R.dimen.margin_compass) + navPadding; - int marginY = UiUtils.dimen(context, R.dimen.margin_compass_top) + navPadding; - nativeSetupWidget(WIDGET_COMPASS, - mWidth - x - marginX, - y + marginY, - ANCHOR_CENTER); - if (forceRedraw && mSurfaceCreated) - nativeApplyWidgets(); - mCurrentCompassOffsetX = x; - mCurrentCompassOffsetY = y; + mMap.setupBottomWidgetsOffset(requireContext(), offsetX, offsetY); } - /** - * Moves the ruler and copyright using the given offsets. - * - * @param offsetY Pixel offset from the bottom. -1 to keep the previous value. - * @param offsetX Pixel offset from the left. -1 to keep the previous value. - */ - void setupBottomWidgetsOffset(int offsetY, int offsetX) + public void destroySurface() { - int x = offsetX < 0 ? mBottomWidgetOffsetX : offsetX; - int y = offsetY < 0 ? mBottomWidgetOffsetY : offsetY; - setupRuler(y, x,true); - setupAttribution(y, x, true); - mBottomWidgetOffsetX = x; - mBottomWidgetOffsetY = y; + mMap.onSurfaceDestroyed(requireActivity().isChangingConfigurations(), isAdded()); } - private void setupRuler(int offsetY, int offsetX, boolean forceRedraw) + public boolean isContextCreated() { - Context context = requireContext(); - nativeSetupWidget(WIDGET_RULER, - UiUtils.dimen(context, R.dimen.margin_ruler) + offsetX, - mHeight - UiUtils.dimen(context, R.dimen.margin_ruler) - offsetY, - ANCHOR_LEFT_BOTTOM); - if (forceRedraw && mSurfaceCreated) - nativeApplyWidgets(); - } - - private void setupAttribution(int offsetY, int offsetX, boolean forceRedraw) - { - Context context = requireContext(); - nativeSetupWidget(WIDGET_COPYRIGHT, - UiUtils.dimen(context, R.dimen.margin_ruler) + offsetX, - mHeight - UiUtils.dimen(context, R.dimen.margin_ruler) - offsetY, - ANCHOR_LEFT_BOTTOM); - if (forceRedraw && mSurfaceCreated) - nativeApplyWidgets(); - } - - private void reportUnsupported() - { - new AlertDialog.Builder(requireActivity(), R.style.MwmTheme_AlertDialog) - .setMessage(R.string.unsupported_phone) - .setCancelable(false) - .setPositiveButton(R.string.close, (dlg, which) -> requireActivity().moveTaskToBack(true)) - .show(); + return mMap.isContextCreated(); } @Override public void surfaceCreated(SurfaceHolder surfaceHolder) { - if (isThemeChangingProcess()) - { - Logger.d(TAG, "Activity is being recreated due theme changing, skip 'surfaceCreated' callback"); - return; - } + int densityDpi; - Logger.d(TAG, "surfaceCreated, mSurfaceCreated = " + mSurfaceCreated); - final Surface surface = surfaceHolder.getSurface(); - if (nativeIsEngineCreated()) - { - if (!nativeAttachSurface(surface)) - { - reportUnsupported(); - return; - } - mSurfaceCreated = true; - mSurfaceAttached = true; - mRequireResize = true; - nativeResumeSurfaceRendering(); - return; - } + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) + densityDpi = ConfigurationHelper.getDensityDpi(requireContext().getResources()); + else + densityDpi = getDensityDpiOld(); - mRequireResize = false; - final Rect rect = surfaceHolder.getSurfaceFrame(); - setupWidgets(rect.width(), rect.height()); - - final DisplayMetrics metrics = new DisplayMetrics(); - requireActivity().getWindowManager().getDefaultDisplay().getMetrics(metrics); - final float exactDensityDpi = metrics.densityDpi; - - final boolean firstStart = LocationHelper.INSTANCE.isInFirstRun(); - if (!nativeCreateEngine(surface, (int) exactDensityDpi, firstStart, mLaunchByDeepLink, - app.organicmaps.BuildConfig.VERSION_CODE)) - { - reportUnsupported(); - return; - } - - if (firstStart) - { - UiThread.runLater(new Runnable() - { - @Override - public void run() - { - LocationHelper.INSTANCE.onExitFromFirstRun(); - } - }); - } - - mSurfaceCreated = true; - mSurfaceAttached = true; - nativeResumeSurfaceRendering(); - if (mMapRenderingListener != null) - mMapRenderingListener.onRenderingCreated(); + mMap.onSurfaceCreated(requireContext(), surfaceHolder.getSurface(), surfaceHolder.getSurfaceFrame(), densityDpi); } @Override public void surfaceChanged(SurfaceHolder surfaceHolder, int format, int width, int height) { - if (isThemeChangingProcess()) - { - Logger.d(TAG, "Activity is being recreated due theme changing, skip 'surfaceChanged' callback"); - return; - } - - Logger.d(TAG, "surfaceChanged, mSurfaceCreated = " + mSurfaceCreated); - if (!mSurfaceCreated || (!mRequireResize && surfaceHolder.isCreating())) - return; - - final Surface surface = surfaceHolder.getSurface(); - nativeSurfaceChanged(surface, width, height); - - mRequireResize = false; - setupWidgets(width, height); - nativeApplyWidgets(); - if (mMapRenderingListener != null) - mMapRenderingListener.onRenderingRestored(); + mMap.onSurfaceChanged(requireContext(), surfaceHolder.getSurface(), surfaceHolder.getSurfaceFrame(), surfaceHolder.isCreating()); } @Override @@ -255,94 +69,68 @@ public class MapFragment extends BaseMwmFragment destroySurface(); } - void destroySurface() - { - Logger.d(TAG, "destroySurface, mSurfaceCreated = " + mSurfaceCreated + - ", mSurfaceAttached = " + mSurfaceAttached + ", isAdded = " + isAdded()); - if (!mSurfaceCreated || !mSurfaceAttached || !isAdded()) - return; - - nativeDetachSurface(!requireActivity().isChangingConfigurations()); - mSurfaceCreated = !nativeDestroySurfaceOnDetach(); - mSurfaceAttached = false; - } - @Override public void onAttach(Context context) { super.onAttach(context); - mMapRenderingListener = (MapRenderingListener) context; + mMap.setMapRenderingListener((MapRenderingListener) context); + mMap.setCallbackUnsupported(this::reportUnsupported); } @Override public void onDetach() { super.onDetach(); - mMapRenderingListener = null; + mMap.setMapRenderingListener(null); + mMap.setCallbackUnsupported(null); } @Override public void onCreate(Bundle b) { super.onCreate(b); - mCurrentCompassOffsetX = 0; - mCurrentCompassOffsetY = 0; - mBottomWidgetOffsetX = 0; - mBottomWidgetOffsetY = 0; setRetainInstance(true); + boolean launchByDeepLink = false; Bundle args = getArguments(); if (args != null) - mLaunchByDeepLink = args.getBoolean(ARG_LAUNCH_BY_DEEP_LINK); + launchByDeepLink = args.getBoolean(Map.ARG_LAUNCH_BY_DEEP_LINK); + mMap.onCreate(launchByDeepLink); } @Override public void onStart() { super.onStart(); - nativeSetRenderingInitializationFinishedListener(mMapRenderingListener); + mMap.onStart(); Logger.d(TAG, "onStart"); } public void onStop() { super.onStop(); - nativeSetRenderingInitializationFinishedListener(null); + mMap.onStop(); Logger.d(TAG, "onStop"); } - private boolean isThemeChangingProcess() - { - return mUiThemeOnPause != null && - !mUiThemeOnPause.equals(Config.getCurrentUiTheme(requireContext())); - } - @Override public void onPause() { - mUiThemeOnPause = Config.getCurrentUiTheme(requireContext()); - - // Pause/Resume can be called without surface creation/destroy. - if (mSurfaceAttached) - nativePauseSurfaceRendering(); - super.onPause(); + mMap.onPause(requireContext()); } @Override public void onResume() { super.onResume(); - - // Pause/Resume can be called without surface creation/destroy. - if (mSurfaceAttached) - nativeResumeSurfaceRendering(); + mMap.onResume(); } @Override public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { View view = inflater.inflate(R.layout.fragment_map, container, false); - mSurfaceView = view.findViewById(R.id.map_surfaceview); + final SurfaceView mSurfaceView = view.findViewById(R.id.map_surfaceview); mSurfaceView.getHolder().addCallback(this); return view; } @@ -350,75 +138,50 @@ public class MapFragment extends BaseMwmFragment @Override public boolean onTouch(View view, MotionEvent event) { - final int count = event.getPointerCount(); - - if (count == 0) - return false; - int action = event.getActionMasked(); int pointerIndex = event.getActionIndex(); switch (action) { - case MotionEvent.ACTION_POINTER_UP: - action = NATIVE_ACTION_UP; - break; - case MotionEvent.ACTION_UP: - action = NATIVE_ACTION_UP; - pointerIndex = 0; - break; - case MotionEvent.ACTION_POINTER_DOWN: - action = NATIVE_ACTION_DOWN; - break; - case MotionEvent.ACTION_DOWN: - action = NATIVE_ACTION_DOWN; - pointerIndex = 0; - break; - case MotionEvent.ACTION_MOVE: - action = NATIVE_ACTION_MOVE; - pointerIndex = INVALID_POINTER_MASK; - break; - case MotionEvent.ACTION_CANCEL: - action = NATIVE_ACTION_CANCEL; - break; - } - - switch (count) - { - case 1: - nativeOnTouch(action, event.getPointerId(0), event.getX(), event.getY(), INVALID_TOUCH_ID, 0, 0, 0); - return true; - default: - nativeOnTouch(action, - event.getPointerId(0), event.getX(0), event.getY(0), - event.getPointerId(1), event.getX(1), event.getY(1), pointerIndex); - return true; + case MotionEvent.ACTION_POINTER_UP: + action = Map.NATIVE_ACTION_UP; + break; + case MotionEvent.ACTION_UP: + action = Map.NATIVE_ACTION_UP; + pointerIndex = 0; + break; + case MotionEvent.ACTION_POINTER_DOWN: + action = Map.NATIVE_ACTION_DOWN; + break; + case MotionEvent.ACTION_DOWN: + action = Map.NATIVE_ACTION_DOWN; + pointerIndex = 0; + break; + case MotionEvent.ACTION_MOVE: + action = Map.NATIVE_ACTION_MOVE; + pointerIndex = Map.INVALID_POINTER_MASK; + break; + case MotionEvent.ACTION_CANCEL: + action = Map.NATIVE_ACTION_CANCEL; + break; } + Map.onTouch(action, event, pointerIndex); + return true; } - boolean isContextCreated() + private void reportUnsupported() { - return mSurfaceCreated; + new AlertDialog.Builder(requireContext(), R.style.MwmTheme_AlertDialog) + .setMessage(R.string.unsupported_phone) + .setCancelable(false) + .setPositiveButton(R.string.close, (dlg, which) -> requireActivity().moveTaskToBack(true)) + .show(); } - static native void nativeCompassUpdated(double north, boolean forceRedraw); - static native void nativeScalePlus(); - static native void nativeScaleMinus(); - public static native boolean nativeShowMapForUrl(String url); - public static native boolean nativeIsEngineCreated(); - static native boolean nativeDestroySurfaceOnDetach(); - private static native boolean nativeCreateEngine(Surface surface, int density, - boolean firstLaunch, - boolean isLaunchByDeepLink, - int appVersionCode); - private static native boolean nativeAttachSurface(Surface surface); - private static native void nativeDetachSurface(boolean destroySurface); - private static native void nativePauseSurfaceRendering(); - private static native void nativeResumeSurfaceRendering(); - private static native void nativeSurfaceChanged(Surface surface, int w, int h); - private static native void nativeOnTouch(int actionType, int id1, float x1, float y1, int id2, float x2, float y2, int maskedPointer); - private static native void nativeSetupWidget(int widget, float x, float y, int anchor); - private static native void nativeApplyWidgets(); - private static native void nativeCleanWidgets(); - private static native void nativeSetRenderingInitializationFinishedListener( - @Nullable MapRenderingListener listener); + @SuppressWarnings("deprecation") + private int getDensityDpiOld() + { + final DisplayMetrics metrics = new DisplayMetrics(); + requireActivity().getWindowManager().getDefaultDisplay().getMetrics(metrics); + return metrics.densityDpi; + } } diff --git a/android/src/app/organicmaps/MwmActivity.java b/android/src/app/organicmaps/MwmActivity.java index 388b53f5c8..b1c92d283f 100644 --- a/android/src/app/organicmaps/MwmActivity.java +++ b/android/src/app/organicmaps/MwmActivity.java @@ -585,7 +585,7 @@ public class MwmActivity extends BaseMwmFragmentActivity if (mMapFragment == null) { Bundle args = new Bundle(); - args.putBoolean(MapFragment.ARG_LAUNCH_BY_DEEP_LINK, isLaunchByDeepLink); + args.putBoolean(Map.ARG_LAUNCH_BY_DEEP_LINK, isLaunchByDeepLink); final FragmentFactory factory = manager.getFragmentFactory(); mMapFragment = (MapFragment) factory.instantiate(getClassLoader(), MapFragment.class.getName()); mMapFragment.setArguments(args); @@ -640,10 +640,10 @@ public class MwmActivity extends BaseMwmFragmentActivity switch (button) { case zoomIn: - MapFragment.nativeScalePlus(); + Map.zoomIn(); break; case zoomOut: - MapFragment.nativeScaleMinus(); + Map.zoomOut(); break; case myPosition: /// @todo Is calls order important here? Should we call onLocationButtonClicked on else branch? @@ -981,7 +981,7 @@ public class MwmActivity extends BaseMwmFragmentActivity private boolean isMapRendererActive() { - return mMapFragment != null && MapFragment.nativeIsEngineCreated() + return mMapFragment != null && Map.isEngineCreated() && mMapFragment.isContextCreated(); } @@ -1219,11 +1219,11 @@ public class MwmActivity extends BaseMwmFragmentActivity if (mMapFragment == null || !mMapFragment.isAdded()) return; - mMapFragment.setupCompass(offsetY, offsetX, true); + mMapFragment.adjustCompass(offsetX, offsetY); CompassData compass = LocationHelper.INSTANCE.getCompassData(); if (compass != null) - MapFragment.nativeCompassUpdated(compass.getNorth(), true); + Map.onCompassUpdated(compass.getNorth(), true); } public void adjustBottomWidgets() @@ -1245,7 +1245,7 @@ public class MwmActivity extends BaseMwmFragmentActivity final int y = Math.max(Math.max(mapButtonsHeight, mainMenuHeight), mNavBarHeight); - mMapFragment.setupBottomWidgetsOffset(y, offsetX); + mMapFragment.adjustBottomWidgets(offsetX, y); } @Override @@ -1627,7 +1627,7 @@ public class MwmActivity extends BaseMwmFragmentActivity @Override public void onCompassUpdated(@NonNull CompassData compass) { - MapFragment.nativeCompassUpdated(compass.getNorth(), false); + Map.onCompassUpdated(compass.getNorth(), false); mNavigationController.updateNorth(); } @@ -1715,10 +1715,10 @@ public class MwmActivity extends BaseMwmFragmentActivity switch (keyCode) { case KeyEvent.KEYCODE_DPAD_DOWN: - MapFragment.nativeScaleMinus(); + Map.zoomOut(); return true; case KeyEvent.KEYCODE_DPAD_UP: - MapFragment.nativeScalePlus(); + Map.zoomIn(); return true; case KeyEvent.KEYCODE_ESCAPE: Intent currIntent = getIntent(); diff --git a/android/src/app/organicmaps/intent/Factory.java b/android/src/app/organicmaps/intent/Factory.java index d5887301b2..5da9d96b59 100644 --- a/android/src/app/organicmaps/intent/Factory.java +++ b/android/src/app/organicmaps/intent/Factory.java @@ -13,6 +13,7 @@ import androidx.fragment.app.Fragment; import androidx.fragment.app.FragmentManager; import app.organicmaps.DownloadResourcesLegacyActivity; import app.organicmaps.Framework; +import app.organicmaps.Map; import app.organicmaps.MapFragment; import app.organicmaps.MwmActivity; import app.organicmaps.MwmApplication; @@ -182,7 +183,7 @@ public class Factory @Override public boolean run(@NonNull MwmActivity target) { - return MapFragment.nativeShowMapForUrl(getUrl()); + return Map.showMapForUrl(getUrl()); } } @@ -203,7 +204,7 @@ public class Factory // TODO: Kernel recognizes "mapsme://", "mwm://" and "mapswithme://" schemas only!!! if (result.getUrlType() == ParsingResult.TYPE_INCORRECT) - return MapFragment.nativeShowMapForUrl(getUrl()); + return Map.showMapForUrl(getUrl()); if (!result.isSuccess()) return false; @@ -223,7 +224,7 @@ public class Factory return false; case ParsingResult.TYPE_MAP: - return MapFragment.nativeShowMapForUrl(getUrl()); + return Map.showMapForUrl(getUrl()); case ParsingResult.TYPE_ROUTE: final ParsedRoutingData data = Framework.nativeGetParsedRoutingData();