blob: 12d16049d1c3201e106d902fb8363d6de5951e80 [file] [log] [blame]
Aurimas Liutikas93554f22022-04-19 16:51:35 -07001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.app;
18
19import android.compat.annotation.UnsupportedAppUsage;
20import android.content.Context;
21import android.content.pm.ActivityInfo;
22import android.content.pm.PackageManager;
23import android.content.res.AssetManager;
24import android.content.res.Configuration;
25import android.graphics.PixelFormat;
26import android.os.Build;
27import android.os.Bundle;
28import android.os.Looper;
29import android.os.MessageQueue;
30import android.util.AttributeSet;
31import android.view.InputQueue;
32import android.view.Surface;
33import android.view.SurfaceHolder;
34import android.view.View;
35import android.view.ViewTreeObserver.OnGlobalLayoutListener;
36import android.view.WindowManager;
37import android.view.inputmethod.InputMethodManager;
38
39import dalvik.system.BaseDexClassLoader;
40
41import java.io.File;
42
43/**
44 * Convenience for implementing an activity that will be implemented
45 * purely in native code. That is, a game (or game-like thing). There
46 * is no need to derive from this class; you can simply declare it in your
47 * manifest, and use the NDK APIs from there.
48 *
49 * <p>A <a href="https://github.com/googlesamples/android-ndk/tree/master/native-activity">sample
50 * native activity</a> is available in the NDK samples.
51 */
52public class NativeActivity extends Activity implements SurfaceHolder.Callback2,
53 InputQueue.Callback, OnGlobalLayoutListener {
54 /**
55 * Optional meta-that can be in the manifest for this component, specifying
56 * the name of the native shared library to load. If not specified,
57 * "main" is used.
58 */
59 public static final String META_DATA_LIB_NAME = "android.app.lib_name";
60
61 /**
62 * Optional meta-that can be in the manifest for this component, specifying
63 * the name of the main entry point for this native activity in the
64 * {@link #META_DATA_LIB_NAME} native code. If not specified,
65 * "ANativeActivity_onCreate" is used.
66 */
67 public static final String META_DATA_FUNC_NAME = "android.app.func_name";
68
69 private static final String KEY_NATIVE_SAVED_STATE = "android:native_state";
70
71 private NativeContentView mNativeContentView;
72 private InputMethodManager mIMM;
73
74 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
75 private long mNativeHandle;
76
77 private InputQueue mCurInputQueue;
78 private SurfaceHolder mCurSurfaceHolder;
79
80 final int[] mLocation = new int[2];
81 int mLastContentX;
82 int mLastContentY;
83 int mLastContentWidth;
84 int mLastContentHeight;
85
86 private boolean mDispatchingUnhandledKey;
87
88 private boolean mDestroyed;
89
90 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
91 private native long loadNativeCode(String path, String funcname, MessageQueue queue,
92 String internalDataPath, String obbPath, String externalDataPath, int sdkVersion,
93 AssetManager assetMgr, byte[] savedState, ClassLoader classLoader, String libraryPath);
94 private native String getDlError();
95 private native void unloadNativeCode(long handle);
96 private native void onStartNative(long handle);
97 private native void onResumeNative(long handle);
98 private native byte[] onSaveInstanceStateNative(long handle);
99 private native void onPauseNative(long handle);
100 private native void onStopNative(long handle);
101 private native void onConfigurationChangedNative(long handle);
102 private native void onLowMemoryNative(long handle);
103 private native void onWindowFocusChangedNative(long handle, boolean focused);
104 private native void onSurfaceCreatedNative(long handle, Surface surface);
105 private native void onSurfaceChangedNative(long handle, Surface surface,
106 int format, int width, int height);
107 private native void onSurfaceRedrawNeededNative(long handle, Surface surface);
108 private native void onSurfaceDestroyedNative(long handle);
109 private native void onInputQueueCreatedNative(long handle, long queuePtr);
110 private native void onInputQueueDestroyedNative(long handle, long queuePtr);
111 private native void onContentRectChangedNative(long handle, int x, int y, int w, int h);
112
113 static class NativeContentView extends View {
114 NativeActivity mActivity;
115
116 public NativeContentView(Context context) {
117 super(context);
118 }
119
120 public NativeContentView(Context context, AttributeSet attrs) {
121 super(context, attrs);
122 }
123 }
124
125 @Override
126 protected void onCreate(Bundle savedInstanceState) {
127 String libname = "main";
128 String funcname = "ANativeActivity_onCreate";
129 ActivityInfo ai;
130
131 mIMM = getSystemService(InputMethodManager.class);
132
133 getWindow().takeSurface(this);
134 getWindow().takeInputQueue(this);
135 getWindow().setFormat(PixelFormat.RGB_565);
136 getWindow().setSoftInputMode(
137 WindowManager.LayoutParams.SOFT_INPUT_STATE_UNSPECIFIED
138 | WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE);
139
140 mNativeContentView = new NativeContentView(this);
141 mNativeContentView.mActivity = this;
142 setContentView(mNativeContentView);
143 mNativeContentView.requestFocus();
144 mNativeContentView.getViewTreeObserver().addOnGlobalLayoutListener(this);
145
146 try {
147 ai = getPackageManager().getActivityInfo(
148 getIntent().getComponent(), PackageManager.GET_META_DATA);
149 if (ai.metaData != null) {
150 String ln = ai.metaData.getString(META_DATA_LIB_NAME);
151 if (ln != null) libname = ln;
152 ln = ai.metaData.getString(META_DATA_FUNC_NAME);
153 if (ln != null) funcname = ln;
154 }
155 } catch (PackageManager.NameNotFoundException e) {
156 throw new RuntimeException("Error getting activity info", e);
157 }
158
159 BaseDexClassLoader classLoader = (BaseDexClassLoader) getClassLoader();
160 String path = classLoader.findLibrary(libname);
161
162 if (path == null) {
163 throw new IllegalArgumentException("Unable to find native library " + libname +
164 " using classloader: " + classLoader.toString());
165 }
166
167 byte[] nativeSavedState = savedInstanceState != null
168 ? savedInstanceState.getByteArray(KEY_NATIVE_SAVED_STATE) : null;
169
170 mNativeHandle = loadNativeCode(path, funcname, Looper.myQueue(),
171 getAbsolutePath(getFilesDir()), getAbsolutePath(getObbDir()),
172 getAbsolutePath(getExternalFilesDir(null)),
173 Build.VERSION.SDK_INT, getAssets(), nativeSavedState,
174 classLoader, classLoader.getLdLibraryPath());
175
176 if (mNativeHandle == 0) {
177 throw new UnsatisfiedLinkError(
178 "Unable to load native library \"" + path + "\": " + getDlError());
179 }
180 super.onCreate(savedInstanceState);
181 }
182
183 private static String getAbsolutePath(File file) {
184 return (file != null) ? file.getAbsolutePath() : null;
185 }
186
187 @Override
188 protected void onDestroy() {
189 mDestroyed = true;
190 if (mCurSurfaceHolder != null) {
191 onSurfaceDestroyedNative(mNativeHandle);
192 mCurSurfaceHolder = null;
193 }
194 if (mCurInputQueue != null) {
195 onInputQueueDestroyedNative(mNativeHandle, mCurInputQueue.getNativePtr());
196 mCurInputQueue = null;
197 }
198 unloadNativeCode(mNativeHandle);
199 super.onDestroy();
200 }
201
202 @Override
203 protected void onPause() {
204 super.onPause();
205 onPauseNative(mNativeHandle);
206 }
207
208 @Override
209 protected void onResume() {
210 super.onResume();
211 onResumeNative(mNativeHandle);
212 }
213
214 @Override
215 protected void onSaveInstanceState(Bundle outState) {
216 super.onSaveInstanceState(outState);
217 byte[] state = onSaveInstanceStateNative(mNativeHandle);
218 if (state != null) {
219 outState.putByteArray(KEY_NATIVE_SAVED_STATE, state);
220 }
221 }
222
223 @Override
224 protected void onStart() {
225 super.onStart();
226 onStartNative(mNativeHandle);
227 }
228
229 @Override
230 protected void onStop() {
231 super.onStop();
232 onStopNative(mNativeHandle);
233 }
234
235 @Override
236 public void onConfigurationChanged(Configuration newConfig) {
237 super.onConfigurationChanged(newConfig);
238 if (!mDestroyed) {
239 onConfigurationChangedNative(mNativeHandle);
240 }
241 }
242
243 @Override
244 public void onLowMemory() {
245 super.onLowMemory();
246 if (!mDestroyed) {
247 onLowMemoryNative(mNativeHandle);
248 }
249 }
250
251 @Override
252 public void onWindowFocusChanged(boolean hasFocus) {
253 super.onWindowFocusChanged(hasFocus);
254 if (!mDestroyed) {
255 onWindowFocusChangedNative(mNativeHandle, hasFocus);
256 }
257 }
258
259 public void surfaceCreated(SurfaceHolder holder) {
260 if (!mDestroyed) {
261 mCurSurfaceHolder = holder;
262 onSurfaceCreatedNative(mNativeHandle, holder.getSurface());
263 }
264 }
265
266 public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
267 if (!mDestroyed) {
268 mCurSurfaceHolder = holder;
269 onSurfaceChangedNative(mNativeHandle, holder.getSurface(), format, width, height);
270 }
271 }
272
273 public void surfaceRedrawNeeded(SurfaceHolder holder) {
274 if (!mDestroyed) {
275 mCurSurfaceHolder = holder;
276 onSurfaceRedrawNeededNative(mNativeHandle, holder.getSurface());
277 }
278 }
279
280 public void surfaceDestroyed(SurfaceHolder holder) {
281 mCurSurfaceHolder = null;
282 if (!mDestroyed) {
283 onSurfaceDestroyedNative(mNativeHandle);
284 }
285 }
286
287 public void onInputQueueCreated(InputQueue queue) {
288 if (!mDestroyed) {
289 mCurInputQueue = queue;
290 onInputQueueCreatedNative(mNativeHandle, queue.getNativePtr());
291 }
292 }
293
294 public void onInputQueueDestroyed(InputQueue queue) {
295 if (!mDestroyed) {
296 onInputQueueDestroyedNative(mNativeHandle, queue.getNativePtr());
297 mCurInputQueue = null;
298 }
299 }
300
301 public void onGlobalLayout() {
302 mNativeContentView.getLocationInWindow(mLocation);
303 int w = mNativeContentView.getWidth();
304 int h = mNativeContentView.getHeight();
305 if (mLocation[0] != mLastContentX || mLocation[1] != mLastContentY
306 || w != mLastContentWidth || h != mLastContentHeight) {
307 mLastContentX = mLocation[0];
308 mLastContentY = mLocation[1];
309 mLastContentWidth = w;
310 mLastContentHeight = h;
311 if (!mDestroyed) {
312 onContentRectChangedNative(mNativeHandle, mLastContentX,
313 mLastContentY, mLastContentWidth, mLastContentHeight);
314 }
315 }
316 }
317
318 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
319 void setWindowFlags(int flags, int mask) {
320 getWindow().setFlags(flags, mask);
321 }
322
323 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
324 void setWindowFormat(int format) {
325 getWindow().setFormat(format);
326 }
327
328 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
329 void showIme(int mode) {
330 mIMM.showSoftInput(mNativeContentView, mode);
331 }
332
333 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
334 void hideIme(int mode) {
335 mIMM.hideSoftInputFromWindow(mNativeContentView.getWindowToken(), mode);
336 }
337}