Touch.java revision 27d377221c7970a5205c83f8a9f62f755dc1fa5d
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2008 The Android Open Source Project 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License"); 59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License. 69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at 79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * http://www.apache.org/licenses/LICENSE-2.0 99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software 119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS, 129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and 149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License. 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.text.method; 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.Layout; 209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.NoCopySpan; 219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.Layout.Alignment; 229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.text.Spannable; 2327d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackbornimport android.util.Log; 249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.MotionEvent; 259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.ViewConfiguration; 269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.widget.TextView; 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class Touch { 299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private Touch() { } 309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Scrolls the specified widget to the specified coordinates, except 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * constrains the X scrolling position to the horizontal regions of 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * the text that will be visible after scrolling to the specified 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Y position. 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void scrollTo(TextView widget, Layout layout, int x, int y) { 389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int padding = widget.getTotalPaddingTop() + 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project widget.getTotalPaddingBottom(); 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int top = layout.getLineForVertical(y); 419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int bottom = layout.getLineForVertical(y + widget.getHeight() - 429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project padding); 439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int left = Integer.MAX_VALUE; 459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int right = 0; 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Alignment a = null; 479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = top; i <= bottom; i++) { 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project left = (int) Math.min(left, layout.getLineLeft(i)); 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project right = (int) Math.max(right, layout.getLineRight(i)); 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (a == null) { 539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project a = layout.getParagraphAlignment(i); 549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project padding = widget.getTotalPaddingLeft() + widget.getTotalPaddingRight(); 589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int width = widget.getWidth(); 599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int diff = 0; 609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (right - left < width - padding) { 629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (a == Alignment.ALIGN_CENTER) { 639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project diff = (width - padding - (right - left)) / 2; 649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (a == Alignment.ALIGN_OPPOSITE) { 659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project diff = width - padding - (right - left); 669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project x = Math.min(x, right - (width - padding) - diff); 709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project x = Math.max(x, left - diff); 719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project widget.scrollTo(x, y); 739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 76eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark * @hide 77eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark * Returns the maximum scroll value in x. 78eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark */ 79eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark public static int getMaxScrollX(TextView widget, Layout layout, int y) { 80eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark int top = layout.getLineForVertical(y); 81eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark int bottom = layout.getLineForVertical(y + widget.getHeight() 82eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark - widget.getTotalPaddingTop() -widget.getTotalPaddingBottom()); 83eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark int left = Integer.MAX_VALUE; 84eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark int right = 0; 85eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark for (int i = top; i <= bottom; i++) { 86eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark left = (int) Math.min(left, layout.getLineLeft(i)); 87eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark right = (int) Math.max(right, layout.getLineRight(i)); 88eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark } 89eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark return right - left - widget.getWidth() - widget.getTotalPaddingLeft() 90eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark - widget.getTotalPaddingRight(); 91eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark } 92eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark 93eaa18dec91b6dd0ce3191a9ab65cdc95ef68b935Cary Clark /** 949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Handles touch events for dragging. You may want to do other actions 959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * like moving the cursor on touch as well. 969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static boolean onTouchEvent(TextView widget, Spannable buffer, 989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project MotionEvent event) { 999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project DragState[] ds; 1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project switch (event.getAction()) { 1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case MotionEvent.ACTION_DOWN: 10358b971d733a2c700cabd3db02b6ea4d5faca6939Eric Fischer ds = buffer.getSpans(0, buffer.length(), DragState.class); 10458b971d733a2c700cabd3db02b6ea4d5faca6939Eric Fischer 10558b971d733a2c700cabd3db02b6ea4d5faca6939Eric Fischer for (int i = 0; i < ds.length; i++) { 10658b971d733a2c700cabd3db02b6ea4d5faca6939Eric Fischer buffer.removeSpan(ds[i]); 10758b971d733a2c700cabd3db02b6ea4d5faca6939Eric Fischer } 10858b971d733a2c700cabd3db02b6ea4d5faca6939Eric Fischer 10938e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn buffer.setSpan(new DragState(event.getX(), event.getY(), 11038e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn widget.getScrollX(), widget.getScrollY()), 11138e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn 0, 0, Spannable.SPAN_MARK_MARK); 1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case MotionEvent.ACTION_UP: 1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ds = buffer.getSpans(0, buffer.length(), DragState.class); 1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < ds.length; i++) { 1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project buffer.removeSpan(ds[i]); 1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ds.length > 0 && ds[0].mUsed) { 1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project case MotionEvent.ACTION_MOVE: 1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ds = buffer.getSpans(0, buffer.length(), DragState.class); 1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ds.length > 0) { 1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ds[0].mFarEnough == false) { 1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int slop = ViewConfiguration.get(widget.getContext()).getScaledTouchSlop(); 1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (Math.abs(event.getX() - ds[0].mX) >= slop || 1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Math.abs(event.getY() - ds[0].mY) >= slop) { 1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ds[0].mFarEnough = true; 1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ds[0].mFarEnough) { 1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ds[0].mUsed = true; 1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project float dx = ds[0].mX - event.getX(); 1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project float dy = ds[0].mY - event.getY(); 1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ds[0].mX = event.getX(); 1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ds[0].mY = event.getY(); 1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int nx = widget.getScrollX() + (int) dx; 1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int ny = widget.getScrollY() + (int) dy; 1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int padding = widget.getTotalPaddingTop() + 1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project widget.getTotalPaddingBottom(); 1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Layout layout = widget.getLayout(); 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ny = Math.min(ny, layout.getHeight() - (widget.getHeight() - 1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project padding)); 1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ny = Math.max(ny, 0); 1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16027d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn int oldX = widget.getScrollX(); 16127d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn int oldY = widget.getScrollY(); 16227d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn 1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project scrollTo(widget, layout, nx, ny); 16427d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn 16527d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn // If we actually scrolled, then cancel the up action. 16627d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn if (oldX != widget.getScrollX() 16727d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn || oldY != widget.getScrollY()) { 16827d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn widget.cancelLongPress(); 16927d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn } 17027d377221c7970a5205c83f8a9f62f755dc1fa5dDianne Hackborn 1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 17938e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn public static int getInitialScrollX(TextView widget, Spannable buffer) { 18038e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn DragState[] ds = buffer.getSpans(0, buffer.length(), DragState.class); 18138e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn return ds.length > 0 ? ds[0].mScrollX : -1; 18238e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn } 18338e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn 18438e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn public static int getInitialScrollY(TextView widget, Spannable buffer) { 18538e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn DragState[] ds = buffer.getSpans(0, buffer.length(), DragState.class); 18638e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn return ds.length > 0 ? ds[0].mScrollY : -1; 18738e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn } 18838e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn 1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static class DragState implements NoCopySpan { 1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public float mX; 1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public float mY; 19238e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn public int mScrollX; 19338e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn public int mScrollY; 1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public boolean mFarEnough; 1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public boolean mUsed; 1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 19738e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn public DragState(float x, float y, int scrollX, int scrollY) { 1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mX = x; 1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mY = y; 20038e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn mScrollX = scrollX; 20138e98fccfab9592f871f3066f8569c559f1ee226Dianne Hackborn mScrollY = scrollY; 2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 205