1 /*
2  * Copyright (C) 2011 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 
17 package android.view;
18 
19 import android.os.Looper;
20 import android.os.MessageQueue;
21 import android.util.Log;
22 import android.util.SparseIntArray;
23 
24 import dalvik.system.CloseGuard;
25 
26 import java.lang.ref.WeakReference;
27 
28 /**
29  * Provides a low-level mechanism for an application to receive input events.
30  * @hide
31  */
32 public abstract class InputEventReceiver {
33     private static final String TAG = "InputEventReceiver";
34 
35     private final CloseGuard mCloseGuard = CloseGuard.get();
36 
37     private long mReceiverPtr;
38 
39     // We keep references to the input channel and message queue objects here so that
40     // they are not GC'd while the native peer of the receiver is using them.
41     private InputChannel mInputChannel;
42     private MessageQueue mMessageQueue;
43 
44     // Map from InputEvent sequence numbers to dispatcher sequence numbers.
45     private final SparseIntArray mSeqMap = new SparseIntArray();
46 
nativeInit(WeakReference<InputEventReceiver> receiver, InputChannel inputChannel, MessageQueue messageQueue)47     private static native long nativeInit(WeakReference<InputEventReceiver> receiver,
48             InputChannel inputChannel, MessageQueue messageQueue);
nativeDispose(long receiverPtr)49     private static native void nativeDispose(long receiverPtr);
nativeFinishInputEvent(long receiverPtr, int seq, boolean handled)50     private static native void nativeFinishInputEvent(long receiverPtr, int seq, boolean handled);
nativeConsumeBatchedInputEvents(long receiverPtr, long frameTimeNanos)51     private static native boolean nativeConsumeBatchedInputEvents(long receiverPtr,
52             long frameTimeNanos);
53 
54     /**
55      * Creates an input event receiver bound to the specified input channel.
56      *
57      * @param inputChannel The input channel.
58      * @param looper The looper to use when invoking callbacks.
59      */
InputEventReceiver(InputChannel inputChannel, Looper looper)60     public InputEventReceiver(InputChannel inputChannel, Looper looper) {
61         if (inputChannel == null) {
62             throw new IllegalArgumentException("inputChannel must not be null");
63         }
64         if (looper == null) {
65             throw new IllegalArgumentException("looper must not be null");
66         }
67 
68         mInputChannel = inputChannel;
69         mMessageQueue = looper.getQueue();
70         mReceiverPtr = nativeInit(new WeakReference<InputEventReceiver>(this),
71                 inputChannel, mMessageQueue);
72 
73         mCloseGuard.open("dispose");
74     }
75 
76     @Override
finalize()77     protected void finalize() throws Throwable {
78         try {
79             dispose(true);
80         } finally {
81             super.finalize();
82         }
83     }
84 
85     /**
86      * Disposes the receiver.
87      */
dispose()88     public void dispose() {
89         dispose(false);
90     }
91 
dispose(boolean finalized)92     private void dispose(boolean finalized) {
93         if (mCloseGuard != null) {
94             if (finalized) {
95                 mCloseGuard.warnIfOpen();
96             }
97             mCloseGuard.close();
98         }
99 
100         if (mReceiverPtr != 0) {
101             nativeDispose(mReceiverPtr);
102             mReceiverPtr = 0;
103         }
104         mInputChannel = null;
105         mMessageQueue = null;
106     }
107 
108     /**
109      * Called when an input event is received.
110      * The recipient should process the input event and then call {@link #finishInputEvent}
111      * to indicate whether the event was handled.  No new input events will be received
112      * until {@link #finishInputEvent} is called.
113      *
114      * @param displayId The display id on which input event triggered.
115      * @param event The input event that was received.
116      */
onInputEvent(InputEvent event, int displayId)117     public void onInputEvent(InputEvent event, int displayId) {
118         finishInputEvent(event, false);
119     }
120 
121     /**
122      * Called when a batched input event is pending.
123      *
124      * The batched input event will continue to accumulate additional movement
125      * samples until the recipient calls {@link #consumeBatchedInputEvents} or
126      * an event is received that ends the batch and causes it to be consumed
127      * immediately (such as a pointer up event).
128      */
onBatchedInputEventPending()129     public void onBatchedInputEventPending() {
130         consumeBatchedInputEvents(-1);
131     }
132 
133     /**
134      * Finishes an input event and indicates whether it was handled.
135      * Must be called on the same Looper thread to which the receiver is attached.
136      *
137      * @param event The input event that was finished.
138      * @param handled True if the event was handled.
139      */
finishInputEvent(InputEvent event, boolean handled)140     public final void finishInputEvent(InputEvent event, boolean handled) {
141         if (event == null) {
142             throw new IllegalArgumentException("event must not be null");
143         }
144         if (mReceiverPtr == 0) {
145             Log.w(TAG, "Attempted to finish an input event but the input event "
146                     + "receiver has already been disposed.");
147         } else {
148             int index = mSeqMap.indexOfKey(event.getSequenceNumber());
149             if (index < 0) {
150                 Log.w(TAG, "Attempted to finish an input event that is not in progress.");
151             } else {
152                 int seq = mSeqMap.valueAt(index);
153                 mSeqMap.removeAt(index);
154                 nativeFinishInputEvent(mReceiverPtr, seq, handled);
155             }
156         }
157         event.recycleIfNeededAfterDispatch();
158     }
159 
160     /**
161      * Consumes all pending batched input events.
162      * Must be called on the same Looper thread to which the receiver is attached.
163      *
164      * This method forces all batched input events to be delivered immediately.
165      * Should be called just before animating or drawing a new frame in the UI.
166      *
167      * @param frameTimeNanos The time in the {@link System#nanoTime()} time base
168      * when the current display frame started rendering, or -1 if unknown.
169      *
170      * @return Whether a batch was consumed
171      */
consumeBatchedInputEvents(long frameTimeNanos)172     public final boolean consumeBatchedInputEvents(long frameTimeNanos) {
173         if (mReceiverPtr == 0) {
174             Log.w(TAG, "Attempted to consume batched input events but the input event "
175                     + "receiver has already been disposed.");
176         } else {
177             return nativeConsumeBatchedInputEvents(mReceiverPtr, frameTimeNanos);
178         }
179         return false;
180     }
181 
182     // Called from native code.
183     @SuppressWarnings("unused")
dispatchInputEvent(int seq, InputEvent event, int displayId)184     private void dispatchInputEvent(int seq, InputEvent event, int displayId) {
185         mSeqMap.put(event.getSequenceNumber(), seq);
186         onInputEvent(event, displayId);
187     }
188 
189     // Called from native code.
190     @SuppressWarnings("unused")
dispatchBatchedInputEventPending()191     private void dispatchBatchedInputEventPending() {
192         onBatchedInputEventPending();
193     }
194 
195     public static interface Factory {
createInputEventReceiver( InputChannel inputChannel, Looper looper)196         public InputEventReceiver createInputEventReceiver(
197                 InputChannel inputChannel, Looper looper);
198     }
199 }
200