1/*
2 * Copyright (C) 2007 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 com.android.commands.input;
18
19import android.hardware.input.InputManager;
20import android.os.SystemClock;
21import android.util.Log;
22import android.view.InputDevice;
23import android.view.KeyCharacterMap;
24import android.view.KeyEvent;
25import android.view.MotionEvent;
26import android.view.ViewConfiguration;
27
28import java.util.HashMap;
29import java.util.Map;
30
31/**
32 * Command that sends key events to the device, either by their keycode, or by
33 * desired character output.
34 */
35
36public class Input {
37    private static final String TAG = "Input";
38    private static final String INVALID_ARGUMENTS = "Error: Invalid arguments for command: ";
39
40    private static final Map<String, Integer> SOURCES = new HashMap<String, Integer>() {{
41        put("keyboard", InputDevice.SOURCE_KEYBOARD);
42        put("dpad", InputDevice.SOURCE_DPAD);
43        put("gamepad", InputDevice.SOURCE_GAMEPAD);
44        put("touchscreen", InputDevice.SOURCE_TOUCHSCREEN);
45        put("mouse", InputDevice.SOURCE_MOUSE);
46        put("stylus", InputDevice.SOURCE_STYLUS);
47        put("trackball", InputDevice.SOURCE_TRACKBALL);
48        put("touchpad", InputDevice.SOURCE_TOUCHPAD);
49        put("touchnavigation", InputDevice.SOURCE_TOUCH_NAVIGATION);
50        put("joystick", InputDevice.SOURCE_JOYSTICK);
51    }};
52
53
54    /**
55     * Command-line entry point.
56     *
57     * @param args The command-line arguments
58     */
59    public static void main(String[] args) {
60        (new Input()).run(args);
61    }
62
63    private void run(String[] args) {
64        if (args.length < 1) {
65            showUsage();
66            return;
67        }
68
69        int index = 0;
70        String command = args[index];
71        int inputSource = InputDevice.SOURCE_UNKNOWN;
72        if (SOURCES.containsKey(command)) {
73            inputSource = SOURCES.get(command);
74            index++;
75            command = args[index];
76        }
77        final int length = args.length - index;
78
79        try {
80            if (command.equals("text")) {
81                if (length == 2) {
82                    inputSource = getSource(inputSource, InputDevice.SOURCE_KEYBOARD);
83                    sendText(inputSource, args[index+1]);
84                    return;
85                }
86            } else if (command.equals("keyevent")) {
87                if (length >= 2) {
88                    final boolean longpress = "--longpress".equals(args[index + 1]);
89                    final int start = longpress ? index + 2 : index + 1;
90                    inputSource = getSource(inputSource, InputDevice.SOURCE_KEYBOARD);
91                    if (args.length > start) {
92                        for (int i = start; i < args.length; i++) {
93                            int keyCode = KeyEvent.keyCodeFromString(args[i]);
94                            if (keyCode == KeyEvent.KEYCODE_UNKNOWN) {
95                                keyCode = KeyEvent.keyCodeFromString("KEYCODE_" + args[i]);
96                            }
97                            sendKeyEvent(inputSource, keyCode, longpress);
98                        }
99                        return;
100                    }
101                }
102            } else if (command.equals("tap")) {
103                if (length == 3) {
104                    inputSource = getSource(inputSource, InputDevice.SOURCE_TOUCHSCREEN);
105                    sendTap(inputSource, Float.parseFloat(args[index+1]),
106                            Float.parseFloat(args[index+2]));
107                    return;
108                }
109            } else if (command.equals("swipe")) {
110                int duration = -1;
111                inputSource = getSource(inputSource, InputDevice.SOURCE_TOUCHSCREEN);
112                switch (length) {
113                    case 6:
114                        duration = Integer.parseInt(args[index+5]);
115                    case 5:
116                        sendSwipe(inputSource,
117                                Float.parseFloat(args[index+1]), Float.parseFloat(args[index+2]),
118                                Float.parseFloat(args[index+3]), Float.parseFloat(args[index+4]),
119                                duration);
120                        return;
121                }
122            } else if (command.equals("draganddrop")) {
123                int duration = -1;
124                inputSource = getSource(inputSource, InputDevice.SOURCE_TOUCHSCREEN);
125                switch (length) {
126                    case 6:
127                        duration = Integer.parseInt(args[index+5]);
128                    case 5:
129                        sendDragAndDrop(inputSource,
130                                Float.parseFloat(args[index+1]), Float.parseFloat(args[index+2]),
131                                Float.parseFloat(args[index+3]), Float.parseFloat(args[index+4]),
132                                duration);
133                        return;
134                }
135            } else if (command.equals("press")) {
136                inputSource = getSource(inputSource, InputDevice.SOURCE_TRACKBALL);
137                if (length == 1) {
138                    sendTap(inputSource, 0.0f, 0.0f);
139                    return;
140                }
141            } else if (command.equals("roll")) {
142                inputSource = getSource(inputSource, InputDevice.SOURCE_TRACKBALL);
143                if (length == 3) {
144                    sendMove(inputSource, Float.parseFloat(args[index+1]),
145                            Float.parseFloat(args[index+2]));
146                    return;
147                }
148            } else {
149                System.err.println("Error: Unknown command: " + command);
150                showUsage();
151                return;
152            }
153        } catch (NumberFormatException ex) {
154        }
155        System.err.println(INVALID_ARGUMENTS + command);
156        showUsage();
157    }
158
159    /**
160     * Convert the characters of string text into key event's and send to
161     * device.
162     *
163     * @param text is a string of characters you want to input to the device.
164     */
165    private void sendText(int source, String text) {
166
167        StringBuffer buff = new StringBuffer(text);
168
169        boolean escapeFlag = false;
170        for (int i=0; i<buff.length(); i++) {
171            if (escapeFlag) {
172                escapeFlag = false;
173                if (buff.charAt(i) == 's') {
174                    buff.setCharAt(i, ' ');
175                    buff.deleteCharAt(--i);
176                }
177            }
178            if (buff.charAt(i) == '%') {
179                escapeFlag = true;
180            }
181        }
182
183        char[] chars = buff.toString().toCharArray();
184
185        KeyCharacterMap kcm = KeyCharacterMap.load(KeyCharacterMap.VIRTUAL_KEYBOARD);
186        KeyEvent[] events = kcm.getEvents(chars);
187        for(int i = 0; i < events.length; i++) {
188            KeyEvent e = events[i];
189            if (source != e.getSource()) {
190                e.setSource(source);
191            }
192            injectKeyEvent(e);
193        }
194    }
195
196    private void sendKeyEvent(int inputSource, int keyCode, boolean longpress) {
197        long now = SystemClock.uptimeMillis();
198        injectKeyEvent(new KeyEvent(now, now, KeyEvent.ACTION_DOWN, keyCode, 0, 0,
199                KeyCharacterMap.VIRTUAL_KEYBOARD, 0, 0, inputSource));
200        if (longpress) {
201            injectKeyEvent(new KeyEvent(now, now, KeyEvent.ACTION_DOWN, keyCode, 1, 0,
202                    KeyCharacterMap.VIRTUAL_KEYBOARD, 0, KeyEvent.FLAG_LONG_PRESS,
203                    inputSource));
204        }
205        injectKeyEvent(new KeyEvent(now, now, KeyEvent.ACTION_UP, keyCode, 0, 0,
206                KeyCharacterMap.VIRTUAL_KEYBOARD, 0, 0, inputSource));
207    }
208
209    private void sendTap(int inputSource, float x, float y) {
210        long now = SystemClock.uptimeMillis();
211        injectMotionEvent(inputSource, MotionEvent.ACTION_DOWN, now, x, y, 1.0f);
212        injectMotionEvent(inputSource, MotionEvent.ACTION_UP, now, x, y, 0.0f);
213    }
214
215    private void sendSwipe(int inputSource, float x1, float y1, float x2, float y2, int duration) {
216        if (duration < 0) {
217            duration = 300;
218        }
219        long now = SystemClock.uptimeMillis();
220        injectMotionEvent(inputSource, MotionEvent.ACTION_DOWN, now, x1, y1, 1.0f);
221        long startTime = now;
222        long endTime = startTime + duration;
223        while (now < endTime) {
224            long elapsedTime = now - startTime;
225            float alpha = (float) elapsedTime / duration;
226            injectMotionEvent(inputSource, MotionEvent.ACTION_MOVE, now, lerp(x1, x2, alpha),
227                    lerp(y1, y2, alpha), 1.0f);
228            now = SystemClock.uptimeMillis();
229        }
230        injectMotionEvent(inputSource, MotionEvent.ACTION_UP, now, x2, y2, 0.0f);
231    }
232
233    private void sendDragAndDrop(int inputSource, float x1, float y1, float x2, float y2,
234            int dragDuration) {
235        if (dragDuration < 0) {
236            dragDuration = 300;
237        }
238        long now = SystemClock.uptimeMillis();
239        injectMotionEvent(inputSource, MotionEvent.ACTION_DOWN, now, x1, y1, 1.0f);
240        try {
241            Thread.sleep(ViewConfiguration.getLongPressTimeout());
242        } catch (InterruptedException e) {
243            throw new RuntimeException(e);
244        }
245        now = SystemClock.uptimeMillis();
246        long startTime = now;
247        long endTime = startTime + dragDuration;
248        while (now < endTime) {
249            long elapsedTime = now - startTime;
250            float alpha = (float) elapsedTime / dragDuration;
251            injectMotionEvent(inputSource, MotionEvent.ACTION_MOVE, now, lerp(x1, x2, alpha),
252                    lerp(y1, y2, alpha), 1.0f);
253            now = SystemClock.uptimeMillis();
254        }
255        injectMotionEvent(inputSource, MotionEvent.ACTION_UP, now, x2, y2, 0.0f);
256    }
257
258    /**
259     * Sends a simple zero-pressure move event.
260     *
261     * @param inputSource the InputDevice.SOURCE_* sending the input event
262     * @param dx change in x coordinate due to move
263     * @param dy change in y coordinate due to move
264     */
265    private void sendMove(int inputSource, float dx, float dy) {
266        long now = SystemClock.uptimeMillis();
267        injectMotionEvent(inputSource, MotionEvent.ACTION_MOVE, now, dx, dy, 0.0f);
268    }
269
270    private void injectKeyEvent(KeyEvent event) {
271        Log.i(TAG, "injectKeyEvent: " + event);
272        InputManager.getInstance().injectInputEvent(event,
273                InputManager.INJECT_INPUT_EVENT_MODE_WAIT_FOR_FINISH);
274    }
275
276    private int getInputDeviceId(int inputSource) {
277        final int DEFAULT_DEVICE_ID = 0;
278        int[] devIds = InputDevice.getDeviceIds();
279        for (int devId : devIds) {
280            InputDevice inputDev = InputDevice.getDevice(devId);
281            if (inputDev.supportsSource(inputSource)) {
282                return devId;
283            }
284        }
285        return DEFAULT_DEVICE_ID;
286    }
287
288    /**
289     * Builds a MotionEvent and injects it into the event stream.
290     *
291     * @param inputSource the InputDevice.SOURCE_* sending the input event
292     * @param action the MotionEvent.ACTION_* for the event
293     * @param when the value of SystemClock.uptimeMillis() at which the event happened
294     * @param x x coordinate of event
295     * @param y y coordinate of event
296     * @param pressure pressure of event
297     */
298    private void injectMotionEvent(int inputSource, int action, long when, float x, float y, float pressure) {
299        final float DEFAULT_SIZE = 1.0f;
300        final int DEFAULT_META_STATE = 0;
301        final float DEFAULT_PRECISION_X = 1.0f;
302        final float DEFAULT_PRECISION_Y = 1.0f;
303        final int DEFAULT_EDGE_FLAGS = 0;
304        MotionEvent event = MotionEvent.obtain(when, when, action, x, y, pressure, DEFAULT_SIZE,
305                DEFAULT_META_STATE, DEFAULT_PRECISION_X, DEFAULT_PRECISION_Y,
306                getInputDeviceId(inputSource), DEFAULT_EDGE_FLAGS);
307        event.setSource(inputSource);
308        Log.i(TAG, "injectMotionEvent: " + event);
309        InputManager.getInstance().injectInputEvent(event,
310                InputManager.INJECT_INPUT_EVENT_MODE_WAIT_FOR_FINISH);
311    }
312
313    private static final float lerp(float a, float b, float alpha) {
314        return (b - a) * alpha + a;
315    }
316
317    private static final int getSource(int inputSource, int defaultSource) {
318        return inputSource == InputDevice.SOURCE_UNKNOWN ? defaultSource : inputSource;
319    }
320
321    private void showUsage() {
322        System.err.println("Usage: input [<source>] <command> [<arg>...]");
323        System.err.println();
324        System.err.println("The sources are: ");
325        for (String src : SOURCES.keySet()) {
326            System.err.println("      " + src);
327        }
328        System.err.println();
329        System.err.println("The commands and default sources are:");
330        System.err.println("      text <string> (Default: touchscreen)");
331        System.err.println("      keyevent [--longpress] <key code number or name> ..."
332                + " (Default: keyboard)");
333        System.err.println("      tap <x> <y> (Default: touchscreen)");
334        System.err.println("      swipe <x1> <y1> <x2> <y2> [duration(ms)]"
335                + " (Default: touchscreen)");
336        System.err.println("      draganddrop <x1> <y1> <x2> <y2> [duration(ms)]"
337                + " (Default: touchscreen)");
338        System.err.println("      press (Default: trackball)");
339        System.err.println("      roll <dx> <dy> (Default: trackball)");
340    }
341}
342