Files
engine/shell/platform/android/io/flutter/view/FlutterView.java
T

925 lines
33 KiB
Java
Raw Normal View History

// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package io.flutter.view;
import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.pm.ApplicationInfo;
import android.content.res.Configuration;
import android.graphics.Bitmap;
2016-01-27 13:20:05 -08:00
import android.graphics.Rect;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Matrix;
import android.graphics.SurfaceTexture;
2015-07-15 11:07:53 -07:00
import android.os.Build;
import android.text.format.DateFormat;
import android.util.AttributeSet;
import android.util.Log;
2016-09-16 14:46:36 -07:00
import android.util.TypedValue;
2015-11-10 16:16:49 -08:00
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.Surface;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
2016-01-27 13:20:05 -08:00
import android.view.WindowInsets;
import android.view.WindowManager;
2016-01-27 13:20:05 -08:00
import android.view.accessibility.AccessibilityManager;
import android.view.accessibility.AccessibilityNodeProvider;
2015-03-09 16:19:56 -07:00
import android.view.inputmethod.EditorInfo;
import android.view.inputmethod.InputConnection;
import io.flutter.app.FlutterActivity;
import io.flutter.plugin.common.*;
import io.flutter.plugin.common.MethodChannel;
2017-03-17 09:04:59 +01:00
import io.flutter.plugin.editing.TextInputPlugin;
import io.flutter.plugin.platform.PlatformPlugin;
import io.flutter.view.VsyncWaiter;
2017-03-17 09:04:59 +01:00
import org.json.JSONException;
import org.json.JSONObject;
2016-11-01 17:11:56 -07:00
import java.net.URI;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.ArrayList;
2017-03-17 09:04:59 +01:00
import java.util.Arrays;
2016-01-27 13:20:05 -08:00
import java.util.HashMap;
import java.util.List;
import java.util.Locale;
2016-01-27 13:20:05 -08:00
import java.util.Map;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
/**
* An Android view containing a Flutter app.
*/
public class FlutterView extends SurfaceView
implements BinaryMessenger, TextureRegistry, AccessibilityManager.AccessibilityStateChangeListener {
2017-03-17 09:04:59 +01:00
/**
* Interface for those objects that maintain and expose a reference to a
* {@code FlutterView} (such as a full-screen Flutter activity).
2017-06-09 07:35:50 -07:00
*
* <p>This indirection is provided to support applications that use an
* activity other than {@link io.flutter.app.FlutterActivity} (e.g. Android
* v4 support library's {@code FragmentActivity}). It allows Flutter plugins
* to deal in this interface and not require that the activity be a subclass
* of {@code FlutterActivity}.</p>
*/
public interface Provider {
/**
* Returns a reference to the Flutter view maintained by this object.
* This may be {@code null}.
*/
FlutterView getFlutterView();
}
private static final String TAG = "FlutterView";
private static final String ACTION_DISCOVER = "io.flutter.view.DISCOVER";
2017-03-17 09:04:59 +01:00
static final class ViewportMetrics {
float devicePixelRatio = 1.0f;
int physicalWidth = 0;
int physicalHeight = 0;
int physicalPaddingTop = 0;
int physicalPaddingRight = 0;
int physicalPaddingBottom = 0;
int physicalPaddingLeft = 0;
}
2017-03-17 09:04:59 +01:00
private final TextInputPlugin mTextInputPlugin;
private final SurfaceHolder.Callback mSurfaceCallback;
private final ViewportMetrics mMetrics;
2016-01-27 13:20:05 -08:00
private final AccessibilityManager mAccessibilityManager;
private final MethodChannel mFlutterLocalizationChannel;
private final MethodChannel mFlutterNavigationChannel;
private final BasicMessageChannel<Object> mFlutterKeyEventChannel;
private final BasicMessageChannel<String> mFlutterLifecycleChannel;
private final BasicMessageChannel<Object> mFlutterSystemChannel;
private final BasicMessageChannel<Object> mFlutterSettingsChannel;
2017-03-17 09:04:59 +01:00
private final BroadcastReceiver mDiscoveryReceiver;
private final List<ActivityLifecycleListener> mActivityLifecycleListeners;
private final List<FirstFrameListener> mFirstFrameListeners;
private final AtomicLong nextTextureId = new AtomicLong(0L);
private FlutterNativeView mNativeView;
private boolean mIsSoftwareRenderingEnabled = false; // using the software renderer or not
public FlutterView(Context context) {
this(context, null);
}
public FlutterView(Context context, AttributeSet attrs) {
2017-11-13 13:56:48 -08:00
this(context, attrs, null);
}
public FlutterView(Context context, AttributeSet attrs, FlutterNativeView nativeView) {
super(context, attrs);
mIsSoftwareRenderingEnabled = nativeGetIsSoftwareRenderingEnabled();
mMetrics = new ViewportMetrics();
mMetrics.devicePixelRatio = context.getResources().getDisplayMetrics().density;
setFocusable(true);
setFocusableInTouchMode(true);
2017-11-13 13:56:48 -08:00
if (nativeView == null) {
mNativeView = new FlutterNativeView(this);
} else {
mNativeView = nativeView;
mNativeView.setFlutterView(this);
}
2016-09-16 14:46:36 -07:00
int color = 0xFF000000;
TypedValue typedValue = new TypedValue();
context.getTheme().resolveAttribute(android.R.attr.colorBackground, typedValue, true);
2017-03-17 09:04:59 +01:00
if (typedValue.type >= TypedValue.TYPE_FIRST_COLOR_INT
&& typedValue.type <= TypedValue.TYPE_LAST_COLOR_INT) {
color = typedValue.data;
}
2016-09-16 14:46:36 -07:00
// TODO(abarth): Consider letting the developer override this color.
final int backgroundColor = color;
2016-09-16 14:46:36 -07:00
mSurfaceCallback = new SurfaceHolder.Callback() {
@Override
public void surfaceCreated(SurfaceHolder holder) {
assertAttached();
nativeSurfaceCreated(mNativeView.get(), holder.getSurface(), backgroundColor);
}
2016-09-16 14:46:36 -07:00
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
assertAttached();
nativeSurfaceChanged(mNativeView.get(), width, height);
2016-09-16 14:46:36 -07:00
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
assertAttached();
nativeSurfaceDestroyed(mNativeView.get());
}
};
getHolder().addCallback(mSurfaceCallback);
2017-03-17 09:04:59 +01:00
mAccessibilityManager = (AccessibilityManager) getContext()
.getSystemService(Context.ACCESSIBILITY_SERVICE);
2017-03-17 09:04:59 +01:00
mActivityLifecycleListeners = new ArrayList<>();
mFirstFrameListeners = new ArrayList<>();
2017-03-17 09:04:59 +01:00
// Configure the platform plugins and flutter channels.
mFlutterLocalizationChannel = new MethodChannel(this, "flutter/localization",
2017-03-17 09:04:59 +01:00
JSONMethodCodec.INSTANCE);
mFlutterNavigationChannel = new MethodChannel(this, "flutter/navigation",
2017-03-17 09:04:59 +01:00
JSONMethodCodec.INSTANCE);
mFlutterKeyEventChannel = new BasicMessageChannel<>(this, "flutter/keyevent",
2017-03-17 09:04:59 +01:00
JSONMessageCodec.INSTANCE);
mFlutterLifecycleChannel = new BasicMessageChannel<>(this, "flutter/lifecycle",
2017-03-17 09:04:59 +01:00
StringCodec.INSTANCE);
mFlutterSystemChannel = new BasicMessageChannel<>(this, "flutter/system",
2017-03-17 09:04:59 +01:00
JSONMessageCodec.INSTANCE);
mFlutterSettingsChannel = new BasicMessageChannel<>(this, "flutter/settings",
JSONMessageCodec.INSTANCE);
// TODO(plugins): Change PlatformPlugin to accept a Context. Disable the
// operations that require an Activity when a Context is passed.
if (getContext() instanceof Activity) {
PlatformPlugin platformPlugin = new PlatformPlugin((Activity) getContext());
MethodChannel flutterPlatformChannel = new MethodChannel(this,
"flutter/platform", JSONMethodCodec.INSTANCE);
flutterPlatformChannel.setMethodCallHandler(platformPlugin);
addActivityLifecycleListener(platformPlugin);
}
mTextInputPlugin = new TextInputPlugin(this);
setLocale(getResources().getConfiguration().locale);
setUserSettings();
if ((context.getApplicationInfo().flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0) {
2017-03-17 09:04:59 +01:00
mDiscoveryReceiver = new DiscoveryReceiver();
context.registerReceiver(mDiscoveryReceiver, new IntentFilter(ACTION_DISCOVER));
} else {
mDiscoveryReceiver = null;
}
2015-11-10 16:16:49 -08:00
}
2017-03-17 09:04:59 +01:00
private void encodeKeyEvent(KeyEvent event, Map<String, Object> message) {
message.put("flags", event.getFlags());
message.put("codePoint", event.getUnicodeChar());
message.put("keyCode", event.getKeyCode());
message.put("scanCode", event.getScanCode());
message.put("metaState", event.getMetaState());
}
2015-11-10 16:16:49 -08:00
@Override
public boolean onKeyUp(int keyCode, KeyEvent event) {
2017-03-17 09:04:59 +01:00
if (!isAttached()) {
return super.onKeyUp(keyCode, event);
}
2017-03-17 09:04:59 +01:00
Map<String, Object> message = new HashMap<>();
message.put("type", "keyup");
message.put("keymap", "android");
encodeKeyEvent(event, message);
mFlutterKeyEventChannel.send(message);
return super.onKeyUp(keyCode, event);
2015-11-10 16:16:49 -08:00
}
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
2017-03-17 09:04:59 +01:00
if (!isAttached()) {
return super.onKeyDown(keyCode, event);
}
2017-03-17 09:04:59 +01:00
Map<String, Object> message = new HashMap<>();
message.put("type", "keydown");
message.put("keymap", "android");
encodeKeyEvent(event, message);
mFlutterKeyEventChannel.send(message);
return super.onKeyDown(keyCode, event);
}
2017-11-13 13:56:48 -08:00
public FlutterNativeView getFlutterNativeView() {
return mNativeView;
}
public void addActivityLifecycleListener(ActivityLifecycleListener listener) {
mActivityLifecycleListeners.add(listener);
}
public void onPause() {
2017-03-17 09:04:59 +01:00
mFlutterLifecycleChannel.send("AppLifecycleState.paused");
}
public void onPostResume() {
2017-03-17 09:04:59 +01:00
for (ActivityLifecycleListener listener : mActivityLifecycleListeners) {
listener.onPostResume();
2017-03-17 09:04:59 +01:00
}
mFlutterLifecycleChannel.send("AppLifecycleState.resumed");
}
public void onStop() {
mFlutterLifecycleChannel.send("AppLifecycleState.suspending");
}
public void onMemoryPressure() {
2017-03-17 09:04:59 +01:00
Map<String, Object> message = new HashMap<>(1);
message.put("type", "memoryPressure");
mFlutterSystemChannel.send(message);
}
/**
* Provide a listener that will be called once when the FlutterView renders its first frame
* to the underlaying SurfaceView.
*/
public void addFirstFrameListener(FirstFrameListener listener) {
mFirstFrameListeners.add(listener);
}
/**
* Remove an existing first frame listener.
*/
public void removeFirstFrameListener(FirstFrameListener listener) {
mFirstFrameListeners.remove(listener);
}
2017-06-06 10:59:41 +02:00
public void setInitialRoute(String route) {
mFlutterNavigationChannel.invokeMethod("setInitialRoute", route);
}
public void pushRoute(String route) {
2017-03-17 09:04:59 +01:00
mFlutterNavigationChannel.invokeMethod("pushRoute", route);
}
public void popRoute() {
2017-03-17 09:04:59 +01:00
mFlutterNavigationChannel.invokeMethod("popRoute", null);
}
private void setUserSettings() {
Map<String, Object> message = new HashMap<>();
message.put("textScaleFactor", getResources().getConfiguration().fontScale);
message.put("alwaysUse24HourFormat", DateFormat.is24HourFormat(getContext()));
mFlutterSettingsChannel.send(message);
}
private void setLocale(Locale locale) {
2017-03-17 09:04:59 +01:00
mFlutterLocalizationChannel.invokeMethod("setLocale",
Arrays.asList(locale.getLanguage(), locale.getCountry()));
}
@Override
protected void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
setLocale(newConfig.locale);
setUserSettings();
}
2016-01-27 13:20:05 -08:00
float getDevicePixelRatio() {
return mMetrics.devicePixelRatio;
}
2017-11-13 13:56:48 -08:00
public FlutterNativeView detach() {
if (!isAttached())
return null;
if (mDiscoveryReceiver != null) {
getContext().unregisterReceiver(mDiscoveryReceiver);
}
getHolder().removeCallback(mSurfaceCallback);
mNativeView.detach();
FlutterNativeView view = mNativeView;
mNativeView = null;
return view;
}
public void destroy() {
if (!isAttached())
return;
2017-03-17 09:04:59 +01:00
if (mDiscoveryReceiver != null) {
getContext().unregisterReceiver(mDiscoveryReceiver);
}
getHolder().removeCallback(mSurfaceCallback);
mNativeView.destroy();
mNativeView = null;
}
2015-03-09 16:19:56 -07:00
@Override
public InputConnection onCreateInputConnection(EditorInfo outAttrs) {
2017-03-17 09:04:59 +01:00
try {
return mTextInputPlugin.createInputConnection(this, outAttrs);
} catch (JSONException e) {
Log.e(TAG, "Failed to create input connection", e);
return null;
}
2015-03-09 16:19:56 -07:00
}
// Must match the PointerChange enum in pointer.dart.
private static final int kPointerChangeCancel = 0;
private static final int kPointerChangeAdd = 1;
private static final int kPointerChangeRemove = 2;
private static final int kPointerChangeHover = 3;
private static final int kPointerChangeDown = 4;
private static final int kPointerChangeMove = 5;
private static final int kPointerChangeUp = 6;
// Must match the PointerDeviceKind enum in pointer.dart.
private static final int kPointerDeviceKindTouch = 0;
private static final int kPointerDeviceKindMouse = 1;
private static final int kPointerDeviceKindStylus = 2;
private static final int kPointerDeviceKindInvertedStylus = 3;
private int getPointerChangeForAction(int maskedAction) {
2015-02-26 12:23:18 -08:00
// Primary pointer:
2015-06-10 16:19:49 -07:00
if (maskedAction == MotionEvent.ACTION_DOWN) {
return kPointerChangeDown;
2015-06-10 16:19:49 -07:00
}
if (maskedAction == MotionEvent.ACTION_UP) {
return kPointerChangeUp;
2015-06-10 16:19:49 -07:00
}
2015-02-26 12:23:18 -08:00
// Secondary pointer:
2015-06-10 16:19:49 -07:00
if (maskedAction == MotionEvent.ACTION_POINTER_DOWN) {
return kPointerChangeDown;
2015-06-10 16:19:49 -07:00
}
if (maskedAction == MotionEvent.ACTION_POINTER_UP) {
return kPointerChangeUp;
2015-06-10 16:19:49 -07:00
}
2015-02-26 12:23:18 -08:00
// All pointers:
2015-06-10 16:19:49 -07:00
if (maskedAction == MotionEvent.ACTION_MOVE) {
return kPointerChangeMove;
2015-06-10 16:19:49 -07:00
}
if (maskedAction == MotionEvent.ACTION_CANCEL) {
return kPointerChangeCancel;
}
return -1;
}
private int getPointerDeviceTypeForToolType(int toolType) {
switch (toolType) {
case MotionEvent.TOOL_TYPE_FINGER:
return kPointerDeviceKindTouch;
case MotionEvent.TOOL_TYPE_STYLUS:
return kPointerDeviceKindStylus;
case MotionEvent.TOOL_TYPE_MOUSE:
return kPointerDeviceKindMouse;
default:
// MotionEvent.TOOL_TYPE_UNKNOWN will reach here.
return -1;
2015-06-10 16:19:49 -07:00
}
}
private void addPointerForIndex(MotionEvent event, int pointerIndex,
2017-03-17 09:04:59 +01:00
ByteBuffer packet) {
int pointerChange = getPointerChangeForAction(event.getActionMasked());
if (pointerChange == -1) {
return;
}
int pointerKind = event.getToolType(pointerIndex);
if (pointerKind == -1) {
return;
}
long timeStamp = event.getEventTime() * 1000; // Convert from milliseconds to microseconds.
packet.putLong(timeStamp); // time_stamp
packet.putLong(pointerChange); // change
packet.putLong(pointerKind); // kind
packet.putLong(event.getPointerId(pointerIndex)); // device
packet.putDouble(event.getX(pointerIndex)); // physical_x
packet.putDouble(event.getY(pointerIndex)); // physical_y
if (pointerKind == kPointerDeviceKindMouse) {
2017-03-17 09:04:59 +01:00
packet.putLong(event.getButtonState() & 0x1F); // buttons
} else if (pointerKind == kPointerDeviceKindStylus) {
2017-03-17 09:04:59 +01:00
packet.putLong((event.getButtonState() >> 4) & 0xF); // buttons
} else {
2017-03-17 09:04:59 +01:00
packet.putLong(0); // buttons
}
packet.putLong(0); // obscured
2015-03-02 12:38:29 -08:00
// TODO(eseidel): Could get the calibrated range if necessary:
// event.getDevice().getMotionRange(MotionEvent.AXIS_PRESSURE)
2017-03-17 09:04:59 +01:00
packet.putDouble(event.getPressure(pointerIndex)); // pressure
packet.putDouble(0.0); // pressure_min
packet.putDouble(1.0); // pressure_max
2015-03-02 12:38:29 -08:00
if (pointerKind == kPointerDeviceKindStylus) {
2017-03-17 09:04:59 +01:00
packet
.putDouble(event.getAxisValue(MotionEvent.AXIS_DISTANCE, pointerIndex)); // distance
packet.putDouble(0.0); // distance_max
} else {
2017-03-17 09:04:59 +01:00
packet.putDouble(0.0); // distance
packet.putDouble(0.0); // distance_max
}
packet.putDouble(event.getToolMajor(pointerIndex)); // radius_major
packet.putDouble(event.getToolMinor(pointerIndex)); // radius_minor
packet.putDouble(0.0); // radius_min
packet.putDouble(0.0); // radius_max
2017-03-17 09:04:59 +01:00
packet.putDouble(
event.getAxisValue(MotionEvent.AXIS_ORIENTATION, pointerIndex)); // orientation
if (pointerKind == kPointerDeviceKindStylus) {
2017-03-17 09:04:59 +01:00
packet.putDouble(event.getAxisValue(MotionEvent.AXIS_TILT, pointerIndex)); // tilt
} else {
2017-03-17 09:04:59 +01:00
packet.putDouble(0.0); // tilt
}
2015-02-26 12:23:18 -08:00
}
@Override
public boolean onTouchEvent(MotionEvent event) {
2017-03-17 09:04:59 +01:00
if (!isAttached()) {
return false;
2017-03-17 09:04:59 +01:00
}
2015-07-15 11:07:53 -07:00
// TODO(abarth): This version check might not be effective in some
// versions of Android that statically compile code and will be upset
// at the lack of |requestUnbufferedDispatch|. Instead, we should factor
// version-dependent code into separate classes for each supported
// version and dispatch dynamically.
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
requestUnbufferedDispatch(event);
}
// These values must match the unpacking code in hooks.dart.
final int kPointerDataFieldCount = 19;
final int kBytePerField = 8;
int pointerCount = event.getPointerCount();
2017-03-17 09:04:59 +01:00
ByteBuffer packet = ByteBuffer
.allocateDirect(pointerCount * kPointerDataFieldCount * kBytePerField);
packet.order(ByteOrder.LITTLE_ENDIAN);
2015-02-26 12:23:18 -08:00
int maskedAction = event.getActionMasked();
// ACTION_UP, ACTION_POINTER_UP, ACTION_DOWN, and ACTION_POINTER_DOWN
// only apply to a single pointer, other events apply to all pointers.
if (maskedAction == MotionEvent.ACTION_UP
2017-03-17 09:04:59 +01:00
|| maskedAction == MotionEvent.ACTION_POINTER_UP
|| maskedAction == MotionEvent.ACTION_DOWN
|| maskedAction == MotionEvent.ACTION_POINTER_DOWN) {
addPointerForIndex(event, event.getActionIndex(), packet);
2015-02-26 12:23:18 -08:00
} else {
// ACTION_MOVE may not actually mean all pointers have moved
// but it's the responsibility of a later part of the system to
// ignore 0-deltas if desired.
for (int p = 0; p < pointerCount; p++) {
addPointerForIndex(event, p, packet);
2015-02-26 12:23:18 -08:00
}
}
assert packet.position() % (kPointerDataFieldCount * kBytePerField) == 0;
nativeDispatchPointerDataPacket(mNativeView.get(), packet, packet.position());
return true;
}
@Override
public boolean onHoverEvent(MotionEvent event) {
2017-03-17 09:04:59 +01:00
if (!isAttached()) {
return false;
2017-03-17 09:04:59 +01:00
}
boolean handled = handleAccessibilityHoverEvent(event);
if (!handled) {
// TODO(ianh): Expose hover events to the platform,
// implementing ADD, REMOVE, etc.
}
return handled;
}
@Override
protected void onSizeChanged(int width, int height, int oldWidth, int oldHeight) {
mMetrics.physicalWidth = width;
mMetrics.physicalHeight = height;
updateViewportMetrics();
super.onSizeChanged(width, height, oldWidth, oldHeight);
}
@Override
public final WindowInsets onApplyWindowInsets(WindowInsets insets) {
mMetrics.physicalPaddingTop = insets.getSystemWindowInsetTop();
mMetrics.physicalPaddingRight = insets.getSystemWindowInsetRight();
mMetrics.physicalPaddingBottom = insets.getSystemWindowInsetBottom();
mMetrics.physicalPaddingLeft = insets.getSystemWindowInsetLeft();
updateViewportMetrics();
return super.onApplyWindowInsets(insets);
}
@Override
@SuppressWarnings("deprecation")
protected boolean fitSystemWindows(Rect insets) {
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.KITKAT) {
mMetrics.physicalPaddingTop = insets.top;
mMetrics.physicalPaddingRight = insets.right;
mMetrics.physicalPaddingBottom = insets.bottom;
mMetrics.physicalPaddingLeft = insets.left;
updateViewportMetrics();
return true;
} else {
return super.fitSystemWindows(insets);
}
}
private boolean isAttached() {
return mNativeView.isAttached();
}
void assertAttached() {
mNativeView.assertAttached();
}
private void preRun() {
2016-01-27 13:20:05 -08:00
resetAccessibilityTree();
}
private void postRun() {
}
2016-10-28 11:46:57 -07:00
public void runFromBundle(String bundlePath, String snapshotOverride) {
runFromBundle(bundlePath, snapshotOverride, "main", false);
2017-10-18 14:19:28 -07:00
}
public void runFromBundle(String bundlePath, String snapshotOverride, String entrypoint) {
runFromBundle(bundlePath, snapshotOverride, entrypoint, false);
}
public void runFromBundle(String bundlePath, String snapshotOverride, String entrypoint, boolean reuseRuntimeController) {
assertAttached();
preRun();
mNativeView.runFromBundle(bundlePath, snapshotOverride, entrypoint, reuseRuntimeController);
postRun();
}
private void runFromSource(final String assetsDirectory, final String main, final String packages) {
2016-08-18 14:00:25 -07:00
Runnable runnable = new Runnable() {
public void run() {
assertAttached();
2016-08-18 14:00:25 -07:00
preRun();
mNativeView.runFromSource(assetsDirectory, main, packages);
2016-08-18 14:00:25 -07:00
postRun();
synchronized (this) {
notify();
}
}
};
try {
synchronized (runnable) {
// Post to the Android UI thread and wait for the response.
post(runnable);
runnable.wait();
}
} catch (InterruptedException e) {
Log.e(TAG, "Thread got interrupted waiting for " +
2017-03-17 09:04:59 +01:00
"RunFromSourceRunnable to finish", e);
2016-08-18 14:00:25 -07:00
}
}
2017-02-03 16:25:25 -08:00
/**
* Return the most recent frame as a bitmap.
2017-03-17 09:04:59 +01:00
*
2017-02-03 16:25:25 -08:00
* @return A bitmap.
*/
public Bitmap getBitmap() {
assertAttached();
return nativeGetBitmap(mNativeView.get());
}
2015-06-10 16:19:49 -07:00
private static native void nativeSurfaceCreated(long nativePlatformViewAndroid,
2017-03-17 09:04:59 +01:00
Surface surface,
int backgroundColor);
private static native void nativeSurfaceChanged(long nativePlatformViewAndroid,
2017-03-17 09:04:59 +01:00
int width,
int height);
2015-06-10 16:19:49 -07:00
private static native void nativeSurfaceDestroyed(long nativePlatformViewAndroid);
2016-10-28 11:46:57 -07:00
2017-03-17 09:04:59 +01:00
private static native void nativeSetViewportMetrics(long nativePlatformViewAndroid,
float devicePixelRatio,
int physicalWidth,
int physicalHeight,
int physicalPaddingTop,
int physicalPaddingRight,
int physicalPaddingBottom,
int physicalPaddingLeft);
2016-10-28 11:46:57 -07:00
private static native Bitmap nativeGetBitmap(long nativePlatformViewAndroid);
2016-01-27 13:20:05 -08:00
2017-03-17 09:04:59 +01:00
private static native void nativeDispatchPointerDataPacket(long nativePlatformViewAndroid,
ByteBuffer buffer, int position);
private static native void nativeDispatchSemanticsAction(long nativePlatformViewAndroid, int id,
int action);
private static native void nativeSetSemanticsEnabled(long nativePlatformViewAndroid,
boolean enabled);
private static native boolean nativeGetIsSoftwareRenderingEnabled();
private static native void nativeRegisterTexture(long nativePlatformViewAndroid, long textureId, SurfaceTexture surfaceTexture);
private static native void nativeMarkTextureFrameAvailable(long nativePlatformViewAndroid, long textureId);
private static native void nativeUnregisterTexture(long nativePlatformViewAndroid, long textureId);
private void updateViewportMetrics() {
if (!isAttached())
return;
nativeSetViewportMetrics(mNativeView.get(),
2017-03-17 09:04:59 +01:00
mMetrics.devicePixelRatio,
mMetrics.physicalWidth,
mMetrics.physicalHeight,
mMetrics.physicalPaddingTop,
mMetrics.physicalPaddingRight,
mMetrics.physicalPaddingBottom,
mMetrics.physicalPaddingLeft);
WindowManager wm = (WindowManager) getContext()
.getSystemService(Context.WINDOW_SERVICE);
float fps = wm.getDefaultDisplay().getRefreshRate();
VsyncWaiter.refreshPeriodNanos = (long)(1000000000.0 / fps);
}
// Called by native to update the semantics/accessibility tree.
public void updateSemantics(ByteBuffer buffer, String[] strings) {
try {
if (mAccessibilityNodeProvider != null) {
buffer.order(ByteOrder.LITTLE_ENDIAN);
mAccessibilityNodeProvider.updateSemantics(buffer, strings);
}
} catch (Exception ex) {
Log.e(TAG, "Uncaught exception while updating semantics", ex);
2016-10-11 13:27:11 -07:00
}
}
// Called by native to notify first Flutter frame rendered.
public void onFirstFrame() {
for (FirstFrameListener listener : mFirstFrameListeners) {
listener.onFirstFrame();
}
}
2016-01-27 13:20:05 -08:00
// ACCESSIBILITY
2016-02-05 10:04:15 -08:00
private boolean mAccessibilityEnabled = false;
private boolean mTouchExplorationEnabled = false;
private TouchExplorationListener mTouchExplorationListener;
protected void dispatchSemanticsAction(int id, int action) {
if (!isAttached())
return;
nativeDispatchSemanticsAction(mNativeView.get(), id, action);
}
2016-01-27 13:20:05 -08:00
@Override
protected void onAttachedToWindow() {
super.onAttachedToWindow();
2016-02-05 10:04:15 -08:00
mAccessibilityEnabled = mAccessibilityManager.isEnabled();
mTouchExplorationEnabled = mAccessibilityManager.isTouchExplorationEnabled();
2017-03-17 09:04:59 +01:00
if (mAccessibilityEnabled || mTouchExplorationEnabled) {
ensureAccessibilityEnabled();
}
2016-02-08 10:11:24 -08:00
resetWillNotDraw();
2016-01-27 13:20:05 -08:00
mAccessibilityManager.addAccessibilityStateChangeListener(this);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
2017-03-17 09:04:59 +01:00
if (mTouchExplorationListener == null) {
mTouchExplorationListener = new TouchExplorationListener();
2017-03-17 09:04:59 +01:00
}
mAccessibilityManager.addTouchExplorationStateChangeListener(mTouchExplorationListener);
}
2016-01-27 13:20:05 -08:00
}
2016-02-08 10:11:24 -08:00
@Override
protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
mAccessibilityManager.removeAccessibilityStateChangeListener(this);
2017-03-17 09:04:59 +01:00
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
mAccessibilityManager
.removeTouchExplorationStateChangeListener(mTouchExplorationListener);
}
2016-02-08 10:11:24 -08:00
}
private void resetWillNotDraw() {
if (!mIsSoftwareRenderingEnabled) {
setWillNotDraw(!(mAccessibilityEnabled || mTouchExplorationEnabled));
} else {
setWillNotDraw(false);
}
2016-02-08 10:11:24 -08:00
}
2016-01-27 13:20:05 -08:00
@Override
public void onAccessibilityStateChanged(boolean enabled) {
2016-02-05 10:04:15 -08:00
if (enabled) {
2016-01-27 13:20:05 -08:00
ensureAccessibilityEnabled();
2016-02-05 10:04:15 -08:00
} else {
mAccessibilityEnabled = false;
2017-05-16 10:31:53 -07:00
if (mAccessibilityNodeProvider != null) {
mAccessibilityNodeProvider.setAccessibilityEnabled(false);
}
2016-02-05 10:04:15 -08:00
}
2016-02-08 10:11:24 -08:00
resetWillNotDraw();
2016-01-27 13:20:05 -08:00
}
class TouchExplorationListener
2017-03-17 09:04:59 +01:00
implements AccessibilityManager.TouchExplorationStateChangeListener {
@Override
public void onTouchExplorationStateChanged(boolean enabled) {
if (enabled) {
mTouchExplorationEnabled = true;
ensureAccessibilityEnabled();
} else {
mTouchExplorationEnabled = false;
if (mAccessibilityNodeProvider != null) {
mAccessibilityNodeProvider.handleTouchExplorationExit();
}
}
resetWillNotDraw();
}
2016-01-27 13:20:05 -08:00
}
@Override
public AccessibilityNodeProvider getAccessibilityNodeProvider() {
ensureAccessibilityEnabled();
return mAccessibilityNodeProvider;
}
private AccessibilityBridge mAccessibilityNodeProvider;
2016-02-04 09:57:33 -08:00
void ensureAccessibilityEnabled() {
if (!isAttached())
return;
2017-05-16 10:31:53 -07:00
mAccessibilityEnabled = true;
2016-02-04 09:57:33 -08:00
if (mAccessibilityNodeProvider == null) {
mAccessibilityNodeProvider = new AccessibilityBridge(this);
nativeSetSemanticsEnabled(mNativeView.get(), true);
2016-02-04 09:57:33 -08:00
}
2017-05-16 10:31:53 -07:00
mAccessibilityNodeProvider.setAccessibilityEnabled(true);
2016-02-04 09:57:33 -08:00
}
void resetAccessibilityTree() {
if (mAccessibilityNodeProvider != null) {
mAccessibilityNodeProvider.reset();
2016-02-04 09:57:33 -08:00
}
}
private boolean handleAccessibilityHoverEvent(MotionEvent event) {
2017-03-17 09:04:59 +01:00
if (!mTouchExplorationEnabled) {
return false;
2017-03-17 09:04:59 +01:00
}
2016-02-08 10:11:24 -08:00
if (event.getAction() == MotionEvent.ACTION_HOVER_ENTER ||
2017-03-17 09:04:59 +01:00
event.getAction() == MotionEvent.ACTION_HOVER_MOVE) {
2016-02-08 10:11:24 -08:00
mAccessibilityNodeProvider.handleTouchExploration(event.getX(), event.getY());
} else if (event.getAction() == MotionEvent.ACTION_HOVER_EXIT) {
mAccessibilityNodeProvider.handleTouchExplorationExit();
} else {
2016-02-08 10:11:24 -08:00
Log.d("flutter", "unexpected accessibility hover event: " + event);
return false;
}
return true;
}
2016-01-27 13:20:05 -08:00
@Override
public void send(String channel, ByteBuffer message) {
mNativeView.send(channel, message);
}
@Override
public void send(String channel, ByteBuffer message, BinaryReply callback) {
mNativeView.send(channel, message, callback);
}
@Override
public void setMessageHandler(String channel, BinaryMessageHandler handler) {
mNativeView.setMessageHandler(channel, handler);
}
2017-03-17 09:04:59 +01:00
/**
* Broadcast receiver used to discover active Flutter instances.
2017-05-16 10:31:53 -07:00
*
* This is used by the `flutter` tool to find the observatory ports
* for all the active Flutter views. We dump the data to the logs
* and the tool scrapes the log lines for the data.
2017-03-17 09:04:59 +01:00
*/
private class DiscoveryReceiver extends BroadcastReceiver {
2017-03-17 09:04:59 +01:00
@Override
public void onReceive(Context context, Intent intent) {
URI observatoryUri = URI.create(FlutterNativeView.getObservatoryUri());
JSONObject discover = new JSONObject();
try {
discover.put("id", getContext().getPackageName());
2016-11-01 17:11:56 -07:00
discover.put("observatoryPort", observatoryUri.getPort());
2017-05-16 10:31:53 -07:00
Log.i(TAG, "DISCOVER: " + discover); // The tool looks for this data. See android_device.dart.
2017-03-17 09:04:59 +01:00
} catch (JSONException e) {
}
}
}
/**
* Listener will be called on the Android UI thread once when Flutter renders the first frame.
*/
public interface FirstFrameListener {
void onFirstFrame();
}
@Override
public TextureRegistry.SurfaceTextureEntry createSurfaceTexture() {
final SurfaceTexture surfaceTexture = new SurfaceTexture(0);
surfaceTexture.detachFromGLContext();
final SurfaceTextureRegistryEntry entry = new SurfaceTextureRegistryEntry(
nextTextureId.getAndIncrement(), surfaceTexture);
nativeRegisterTexture(mNativeView.get(), entry.id(), surfaceTexture);
return entry;
}
final class SurfaceTextureRegistryEntry implements TextureRegistry.SurfaceTextureEntry {
private final long id;
private final SurfaceTexture surfaceTexture;
private boolean released;
SurfaceTextureRegistryEntry(long id, SurfaceTexture surfaceTexture) {
this.id = id;
this.surfaceTexture = surfaceTexture;
this.surfaceTexture.setOnFrameAvailableListener(new SurfaceTexture.OnFrameAvailableListener() {
@Override
public void onFrameAvailable(SurfaceTexture texture) {
nativeMarkTextureFrameAvailable(mNativeView.get(), SurfaceTextureRegistryEntry.this.id);
}
});
}
@Override
public SurfaceTexture surfaceTexture() {
return surfaceTexture;
}
@Override
public long id() {
return id;
}
@Override
public void release() {
if (released) {
return;
}
released = true;
nativeUnregisterTexture(mNativeView.get(), id);
surfaceTexture.release();
}
}
}