19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2006 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; 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Canvas; 20f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Feltimport android.graphics.Paint; 2134ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunneimport android.util.Log; 226435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 236435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunneimport com.android.internal.util.ArrayUtils; 249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.reflect.Array; 269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/** 289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This is the class for text whose content and markup can both be changed. 299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 306435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunnepublic class SpannableStringBuilder implements CharSequence, GetChars, Spannable, Editable, 316435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne Appendable, GraphicsOperations { 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Create a new SpannableStringBuilder with empty contents 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder() { 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project this(""); 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Create a new SpannableStringBuilder containing a copy of the 419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * specified text, including its spans if any. 429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder(CharSequence text) { 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project this(text, 0, text.length()); 459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Create a new SpannableStringBuilder containing a copy of the 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * specified slice of the specified text, including its spans if any. 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder(CharSequence text, int start, int end) { 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int srclen = end - start; 539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 540249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (srclen < 0) throw new StringIndexOutOfBoundsException(); 550249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne 569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int len = ArrayUtils.idealCharArraySize(srclen + 1); 579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mText = new char[len]; 589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mGapStart = srclen; 599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mGapLength = len - srclen; 609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project TextUtils.getChars(text, start, end, mText, 0); 629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanCount = 0; 649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int alloc = ArrayUtils.idealIntArraySize(0); 659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpans = new Object[alloc]; 669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanStarts = new int[alloc]; 679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanEnds = new int[alloc]; 689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanFlags = new int[alloc]; 699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (text instanceof Spanned) { 719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Spanned sp = (Spanned) text; 729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = sp.getSpans(start, end, Object.class); 739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < spans.length; i++) { 759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spans[i] instanceof NoCopySpan) { 769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 78174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int st = sp.getSpanStart(spans[i]) - start; 809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int en = sp.getSpanEnd(spans[i]) - start; 819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int fl = sp.getSpanFlags(spans[i]); 829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (st < 0) 849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project st = 0; 859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (st > end - start) 869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project st = end - start; 879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (en < 0) 899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project en = 0; 909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (en > end - start) 919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project en = end - start; 929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 930249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne setSpan(false, spans[i], st, en, fl); 949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static SpannableStringBuilder valueOf(CharSequence source) { 999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (source instanceof SpannableStringBuilder) { 1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return (SpannableStringBuilder) source; 1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return new SpannableStringBuilder(source); 1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return the char at the specified offset within the buffer. 1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public char charAt(int where) { 1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int len = length(); 1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (where < 0) { 1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IndexOutOfBoundsException("charAt: " + where + " < 0"); 1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (where >= len) { 1146435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne throw new IndexOutOfBoundsException("charAt: " + where + " >= length " + len); 1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (where >= mGapStart) 1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mText[where + mGapLength]; 1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project else 1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mText[where]; 1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return the number of chars in the buffer. 1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int length() { 1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mText.length - mGapLength; 1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private void resizeFor(int size) { 1317c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne final int oldLength = mText.length; 1327c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne final int newLength = ArrayUtils.idealCharArraySize(size + 1); 13390985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne final int delta = newLength - oldLength; 13490985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne if (delta == 0) return; 1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1367c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne char[] newText = new char[newLength]; 1377c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne System.arraycopy(mText, 0, newText, 0, mGapStart); 13890985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne final int after = oldLength - (mGapStart + mGapLength); 1397c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne System.arraycopy(mText, oldLength - after, newText, newLength - after, after); 1407c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne mText = newText; 1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1427c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne mGapLength += delta; 1437c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne if (mGapLength < 1) 1447c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne new Exception("mGapLength < 1").printStackTrace(); 1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < mSpanCount; i++) { 1477c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne if (mSpanStarts[i] > mGapStart) mSpanStarts[i] += delta; 1487c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne if (mSpanEnds[i] > mGapStart) mSpanEnds[i] += delta; 1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private void moveGapTo(int where) { 1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (where == mGapStart) 1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1560249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne boolean atEnd = (where == length()); 1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (where < mGapStart) { 1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int overlap = mGapStart - where; 1607c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne System.arraycopy(mText, where, mText, mGapStart + mGapLength - overlap, overlap); 1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else /* where > mGapStart */ { 1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int overlap = where - mGapStart; 1637c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne System.arraycopy(mText, where + mGapLength - overlap, mText, mGapStart, overlap); 1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // XXX be more clever 1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < mSpanCount; i++) { 1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int start = mSpanStarts[i]; 1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int end = mSpanEnds[i]; 1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (start > mGapStart) 1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project start -= mGapLength; 1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (start > where) 1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project start += mGapLength; 1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project else if (start == where) { 1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int flag = (mSpanFlags[i] & START_MASK) >> START_SHIFT; 1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1780249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flag == POINT || (atEnd && flag == PARAGRAPH)) 1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project start += mGapLength; 1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end > mGapStart) 1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end -= mGapLength; 1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end > where) 1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end += mGapLength; 1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project else if (end == where) { 1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int flag = (mSpanFlags[i] & END_MASK); 1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1890249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flag == POINT || (atEnd && flag == PARAGRAPH)) 1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end += mGapLength; 1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanStarts[i] = start; 1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanEnds[i] = end; 1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mGapStart = where; 1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder insert(int where, CharSequence tb, int start, int end) { 2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return replace(where, where, tb, start, end); 2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder insert(int where, CharSequence tb) { 2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return replace(where, where, tb, 0, tb.length()); 2089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder delete(int start, int end) { 2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project SpannableStringBuilder ret = replace(start, end, "", 0, 0); 2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mGapLength > 2 * length()) 2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project resizeFor(length()); 216174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return ret; // == this 2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void clear() { 2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project replace(0, length(), "", 0, 0); 2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 224174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void clearSpans() { 2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = mSpanCount - 1; i >= 0; i--) { 2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object what = mSpans[i]; 2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int ostart = mSpanStarts[i]; 2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int oend = mSpanEnds[i]; 2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ostart > mGapStart) 2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ostart -= mGapLength; 2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (oend > mGapStart) 2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project oend -= mGapLength; 2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanCount = i; 2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpans[i] = null; 2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sendSpanRemoved(what, ostart, oend); 2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder append(CharSequence text) { 2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int length = length(); 2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return replace(length, length, text, 0, text.length()); 2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder append(CharSequence text, int start, int end) { 2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int length = length(); 2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return replace(length, length, text, start, end); 2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder append(char text) { 2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return append(String.valueOf(text)); 2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 261174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private void change(int start, int end, CharSequence cs, int csStart, int csEnd) { 262174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // Can be negative 26326b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne final int replacedLength = end - start; 26426b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne final int replacementLength = csEnd - csStart; 26526b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne final int nbNewChars = replacementLength - replacedLength; 2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = mSpanCount - 1; i >= 0; i--) { 268174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanStart = mSpanStarts[i]; 269174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart > mGapStart) 270174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanStart -= mGapLength; 2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 272174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanEnd = mSpanEnds[i]; 273174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd > mGapStart) 274174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanEnd -= mGapLength; 2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 276174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if ((mSpanFlags[i] & SPAN_PARAGRAPH) == SPAN_PARAGRAPH) { 277174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int ost = spanStart; 278174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int oen = spanEnd; 2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int clen = length(); 2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 281174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart > start && spanStart <= end) { 282174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne for (spanStart = end; spanStart < clen; spanStart++) 283174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart > end && charAt(spanStart - 1) == '\n') 2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 287174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd > start && spanEnd <= end) { 288174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne for (spanEnd = end; spanEnd < clen; spanEnd++) 289174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd > end && charAt(spanEnd - 1) == '\n') 2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 293174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart != ost || spanEnd != oen) 294174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne setSpan(false, mSpans[i], spanStart, spanEnd, mSpanFlags[i]); 2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 296174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 297174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int flags = 0; 298174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart == start) flags |= SPAN_START_AT_START; 299174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne else if (spanStart == end + nbNewChars) flags |= SPAN_START_AT_END; 300174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd == start) flags |= SPAN_END_AT_START; 301174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne else if (spanEnd == end + nbNewChars) flags |= SPAN_END_AT_END; 302174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne mSpanFlags[i] |= flags; 3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project moveGapTo(end); 3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 307312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne if (nbNewChars >= mGapLength) { 308312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne resizeFor(mText.length + nbNewChars - mGapLength); 309312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne } 310312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne 311e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne final boolean textIsRemoved = replacementLength == 0; 31290985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // The removal pass needs to be done before the gap is updated in order to broadcast the 31390985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // correct previous positions to the correct intersecting SpanWatchers 31426b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (replacedLength > 0) { // no need for span fixup on pure insertion 31590985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // A for loop will not work because the array is being modified 31690985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // Do not iterate in reverse to keep the SpanWatchers notified in ordering 31790985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // Also, a removed SpanWatcher should not get notified of removed spans located 31890985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // further in the span array. 31990985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne int i = 0; 32090985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne while (i < mSpanCount) { 32190985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne if ((mSpanFlags[i] & Spanned.SPAN_EXCLUSIVE_EXCLUSIVE) == 32290985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne Spanned.SPAN_EXCLUSIVE_EXCLUSIVE && 323e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne mSpanStarts[i] >= start && mSpanStarts[i] < mGapStart + mGapLength && 324e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne mSpanEnds[i] >= start && mSpanEnds[i] < mGapStart + mGapLength && 325e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne // This condition indicates that the span would become empty 326e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne (textIsRemoved || mSpanStarts[i] > start || mSpanEnds[i] < mGapStart)) { 32790985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne removeSpan(i); 328e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne continue; // do not increment i, spans will be shifted left in the array 32990985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne } 330e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne 331e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne i++; 33290985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne } 33390985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne } 33490985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne 335312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne mGapStart += nbNewChars; 336312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne mGapLength -= nbNewChars; 3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mGapLength < 1) 3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project new Exception("mGapLength < 1").printStackTrace(); 3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 341174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne TextUtils.getChars(cs, csStart, csEnd, mText, start); 3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 34326b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (replacedLength > 0) { // no need for span fixup on pure insertion 34490985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne final boolean atEnd = (mGapStart + mGapLength == mText.length); 3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 34690985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne for (int i = 0; i < mSpanCount; i++) { 34726b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne final int startFlag = (mSpanFlags[i] & START_MASK) >> START_SHIFT; 34826b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne mSpanStarts[i] = updatedIntervalBound(mSpanStarts[i], start, nbNewChars, startFlag, 34926b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne atEnd, textIsRemoved); 3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 35126b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne final int endFlag = (mSpanFlags[i] & END_MASK); 35226b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne mSpanEnds[i] = updatedIntervalBound(mSpanEnds[i], start, nbNewChars, endFlag, 35326b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne atEnd, textIsRemoved); 3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3560249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne 357174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne mSpanCountBeforeAdd = mSpanCount; 358174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 359174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (cs instanceof Spanned) { 360174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne Spanned sp = (Spanned) cs; 361174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne Object[] spans = sp.getSpans(csStart, csEnd, Object.class); 3620249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne 3630249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne for (int i = 0; i < spans.length; i++) { 3640249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne int st = sp.getSpanStart(spans[i]); 3650249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne int en = sp.getSpanEnd(spans[i]); 3660249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne 367174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (st < csStart) st = csStart; 368174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (en > csEnd) en = csEnd; 3690249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne 3700249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne // Add span only if this object is not yet used as a span in this string 37190985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne if (getSpanStart(spans[i]) < 0) { 372174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne setSpan(false, spans[i], st - csStart + start, en - csStart + start, 373174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne sp.getSpanFlags(spans[i])); 3740249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne } 3750249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne } 3760249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne } 3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 37926b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne private int updatedIntervalBound(int offset, int start, int nbNewChars, int flag, boolean atEnd, 38026b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne boolean textIsRemoved) { 38126b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (offset >= start && offset < mGapStart + mGapLength) { 38226b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (flag == POINT) { 38326b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne // A POINT located inside the replaced range should be moved to the end of the 38426b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne // replaced text. 38526b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne // The exception is when the point is at the start of the range and we are doing a 38626b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne // text replacement (as opposed to a deletion): the point stays there. 38726b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (textIsRemoved || offset > start) { 38826b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne return mGapStart + mGapLength; 38926b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 39026b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } else { 39126b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (flag == PARAGRAPH) { 39226b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (atEnd) { 39326b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne return mGapStart + mGapLength; 39426b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 39526b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } else { // MARK 396e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne // MARKs should be moved to the start, with the exception of a mark located at 397e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne // the end of the range (which will be < mGapStart + mGapLength since mGapLength 398e244868053cfe44742b42e75df3a08d0c5914bcaGilles Debunne // is > 0, which should stay 'unchanged' at the end of the replaced text. 39926b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne if (textIsRemoved || offset < mGapStart - nbNewChars) { 40026b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne return start; 40126b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } else { 40226b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne // Move to the end of replaced text (needed if nbNewChars != 0) 40326b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne return mGapStart; 40426b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 40526b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 40626b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 40726b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 40826b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne return offset; 40926b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne } 41026b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne 41175beb336f4b164c3bed5d4b91f0b9c6ea49a3437Gilles Debunne private void removeSpan(int i) { 4126435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne Object object = mSpans[i]; 4136435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 4146435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne int start = mSpanStarts[i]; 4156435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne int end = mSpanEnds[i]; 4166435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 4176435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne if (start > mGapStart) start -= mGapLength; 4186435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne if (end > mGapStart) end -= mGapLength; 4196435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 4206435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne int count = mSpanCount - (i + 1); 4216435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne System.arraycopy(mSpans, i + 1, mSpans, i, count); 4226435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne System.arraycopy(mSpanStarts, i + 1, mSpanStarts, i, count); 4236435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne System.arraycopy(mSpanEnds, i + 1, mSpanEnds, i, count); 4246435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne System.arraycopy(mSpanFlags, i + 1, mSpanFlags, i, count); 42575beb336f4b164c3bed5d4b91f0b9c6ea49a3437Gilles Debunne 42675beb336f4b164c3bed5d4b91f0b9c6ea49a3437Gilles Debunne mSpanCount--; 4276435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 4286435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne mSpans[mSpanCount] = null; 4296435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 4306435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne sendSpanRemoved(object, start, end); 43175beb336f4b164c3bed5d4b91f0b9c6ea49a3437Gilles Debunne } 43275beb336f4b164c3bed5d4b91f0b9c6ea49a3437Gilles Debunne 4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder replace(int start, int end, CharSequence tb) { 4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return replace(start, end, tb, 0, tb.length()); 4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public SpannableStringBuilder replace(final int start, final int end, 4400249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne CharSequence tb, int tbstart, int tbend) { 441174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne checkRange("replace", start, end); 442174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int filtercount = mFilters.length; 4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < filtercount; i++) { 445b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne CharSequence repl = mFilters[i].filter(tb, tbstart, tbend, this, start, end); 4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (repl != null) { 4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project tb = repl; 4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project tbstart = 0; 4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project tbend = repl.length(); 4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 454b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne final int origLen = end - start; 455b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne final int newLen = tbend - tbstart; 456b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne 457d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne if (origLen == 0 && newLen == 0 && !hasNonExclusiveExclusiveSpanAt(tb, tbstart)) { 458d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne // This is a no-op iif there are no spans in tb that would be added (with a 0-length) 459d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne // Early exit so that the text watchers do not get notified 460d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne return this; 461d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne } 462d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne 463b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne TextWatcher[] textWatchers = getSpans(start, start + origLen, TextWatcher.class); 464b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne sendBeforeTextChanged(textWatchers, start, origLen, newLen); 465b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne 4660249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne // Try to keep the cursor / selection at the same relative position during 4670249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne // a text replacement. If replaced or replacement text length is zero, this 4680249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne // is already taken care of. 4690249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne boolean adjustSelection = origLen != 0 && newLen != 0; 470174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int selectionStart = 0; 471174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int selectionEnd = 0; 4720249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (adjustSelection) { 473174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne selectionStart = Selection.getSelectionStart(this); 474174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne selectionEnd = Selection.getSelectionEnd(this); 4750249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne } 476bc6862300fe5675addfe4ed5d0c7c345aad463aaGilles Debunne 4770249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne change(start, end, tb, tbstart, tbend); 478bc6862300fe5675addfe4ed5d0c7c345aad463aaGilles Debunne 4790249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (adjustSelection) { 480174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (selectionStart > start && selectionStart < end) { 481174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne final int offset = (selectionStart - start) * newLen / origLen; 482174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne selectionStart = start + offset; 4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 484174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne setSpan(false, Selection.SELECTION_START, selectionStart, selectionStart, 4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Spanned.SPAN_POINT_POINT); 4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 487174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (selectionEnd > start && selectionEnd < end) { 488174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne final int offset = (selectionEnd - start) * newLen / origLen; 489174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne selectionEnd = start + offset; 4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 491174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne setSpan(false, Selection.SELECTION_END, selectionEnd, selectionEnd, 492174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne Spanned.SPAN_POINT_POINT); 4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4956435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 496b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne sendTextChanged(textWatchers, start, origLen, newLen); 497b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne sendAfterTextChanged(textWatchers); 498b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne 499174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // Span watchers need to be called after text watchers, which may update the layout 500174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne sendToSpanWatchers(start, end, newLen - origLen); 501174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return this; 5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 505d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne private static boolean hasNonExclusiveExclusiveSpanAt(CharSequence text, int offset) { 506d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne if (text instanceof Spanned) { 507d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne Spanned spanned = (Spanned) text; 508d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne Object[] spans = spanned.getSpans(offset, offset, Object.class); 509d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne final int length = spans.length; 510d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne for (int i = 0; i < length; i++) { 511d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne Object span = spans[i]; 512d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne int flags = spanned.getSpanFlags(span); 513d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne if (flags != Spanned.SPAN_EXCLUSIVE_EXCLUSIVE) return true; 514d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne } 515d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne } 516d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne return false; 517d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne } 518d60da058c6a9a63ef7347685f39a9dedebbc6afaGilles Debunne 519174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private void sendToSpanWatchers(int replaceStart, int replaceEnd, int nbNewChars) { 520174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne for (int i = 0; i < mSpanCountBeforeAdd; i++) { 521174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanStart = mSpanStarts[i]; 522174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanEnd = mSpanEnds[i]; 523174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart > mGapStart) spanStart -= mGapLength; 524174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd > mGapStart) spanEnd -= mGapLength; 525174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanFlags = mSpanFlags[i]; 526174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 527174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int newReplaceEnd = replaceEnd + nbNewChars; 528174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne boolean spanChanged = false; 52990985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne 530174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int previousSpanStart = spanStart; 531174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart > newReplaceEnd) { 532174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (nbNewChars != 0) { 533174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne previousSpanStart -= nbNewChars; 534174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanChanged = true; 535174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 536174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } else if (spanStart >= replaceStart) { 537174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // No change if span start was already at replace interval boundaries before replace 538174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if ((spanStart != replaceStart || 539174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne ((spanFlags & SPAN_START_AT_START) != SPAN_START_AT_START)) && 540174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne (spanStart != newReplaceEnd || 541174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne ((spanFlags & SPAN_START_AT_END) != SPAN_START_AT_END))) { 54290985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // TODO A correct previousSpanStart cannot be computed at this point. 54390985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // It would require to save all the previous spans' positions before the replace 54490985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne // Using an invalid -1 value to convey this would break the broacast range 545174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanChanged = true; 546174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 547174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 54890985286442340b9ad9433d6bf8d51702c8d0fd9Gilles Debunne 549174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int previousSpanEnd = spanEnd; 550174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd > newReplaceEnd) { 551174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (nbNewChars != 0) { 552174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne previousSpanEnd -= nbNewChars; 553174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanChanged = true; 554174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 555174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } else if (spanEnd >= replaceStart) { 556174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // No change if span start was already at replace interval boundaries before replace 557174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if ((spanEnd != replaceStart || 558174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne ((spanFlags & SPAN_END_AT_START) != SPAN_END_AT_START)) && 559174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne (spanEnd != newReplaceEnd || 560174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne ((spanFlags & SPAN_END_AT_END) != SPAN_END_AT_END))) { 561174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // TODO same as above for previousSpanEnd 562174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanChanged = true; 563174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 564174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 565174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 566174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanChanged) { 567174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne sendSpanChanged(mSpans[i], previousSpanStart, previousSpanEnd, spanStart, spanEnd); 568174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 569174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne mSpanFlags[i] &= ~SPAN_START_END_MASK; 570174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 571174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 572174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // The spans starting at mIntermediateSpanCount were added from the replacement text 573174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne for (int i = mSpanCountBeforeAdd; i < mSpanCount; i++) { 574174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanStart = mSpanStarts[i]; 575174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int spanEnd = mSpanEnds[i]; 576174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanStart > mGapStart) spanStart -= mGapLength; 577174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne if (spanEnd > mGapStart) spanEnd -= mGapLength; 578174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne sendSpanAdded(mSpans[i], spanStart, spanEnd); 579174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 580174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne } 581174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Mark the specified range of text with the specified object. 5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The flags determine how the span will behave when text is 5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * inserted at the start or end of the span's range. 5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void setSpan(Object what, int start, int end, int flags) { 5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project setSpan(true, what, start, end, flags); 5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5916435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne private void setSpan(boolean send, Object what, int start, int end, int flags) { 5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project checkRange("setSpan", start, end); 5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5940249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne int flagsStart = (flags & START_MASK) >> START_SHIFT; 5950249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flagsStart == PARAGRAPH) { 5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (start != 0 && start != length()) { 5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char c = charAt(start - 1); 5989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (c != '\n') 6006435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne throw new RuntimeException("PARAGRAPH span must start at paragraph boundary"); 6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6040249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne int flagsEnd = flags & END_MASK; 6050249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flagsEnd == PARAGRAPH) { 6069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end != 0 && end != length()) { 6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char c = charAt(end - 1); 6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (c != '\n') 6106435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne throw new RuntimeException("PARAGRAPH span must end at paragraph boundary"); 6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6140249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne // 0-length Spanned.SPAN_EXCLUSIVE_EXCLUSIVE 6150249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flagsStart == POINT && flagsEnd == MARK && start == end) { 61634ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunne if (send) Log.e("SpannableStringBuilder", 61734ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunne "SPAN_EXCLUSIVE_EXCLUSIVE spans cannot have a zero length"); 61834ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunne // Silently ignore invalid spans when they are created from this class. 61934ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunne // This avoids the duplication of the above test code before all the 62034ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunne // calls to setSpan that are done in this class 62134ec2b54da52eeed39d9cf9e5f215650b447edd7Gilles Debunne return; 6227c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne } 6237c5f670bd698c984dcafea8125f4d3939bc42972Gilles Debunne 6240249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne int nstart = start; 6250249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne int nend = end; 6260249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne 6276435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne if (start > mGapStart) { 6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project start += mGapLength; 6296435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne } else if (start == mGapStart) { 6300249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flagsStart == POINT || (flagsStart == PARAGRAPH && start == length())) 6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project start += mGapLength; 6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6346435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne if (end > mGapStart) { 6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end += mGapLength; 6366435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne } else if (end == mGapStart) { 6370249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne if (flagsEnd == POINT || (flagsEnd == PARAGRAPH && end == length())) 6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project end += mGapLength; 6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = mSpanCount; 6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = mSpans; 6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spans[i] == what) { 6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int ostart = mSpanStarts[i]; 6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int oend = mSpanEnds[i]; 6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ostart > mGapStart) 6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ostart -= mGapLength; 6519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (oend > mGapStart) 6529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project oend -= mGapLength; 6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanStarts[i] = start; 6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanEnds[i] = end; 6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanFlags[i] = flags; 6579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 658fc1190b8f143bd00c825d4e52928c577e40e6c49Gilles Debunne if (send) sendSpanChanged(what, ostart, oend, nstart, nend); 6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mSpanCount + 1 >= mSpans.length) { 6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int newsize = ArrayUtils.idealIntArraySize(mSpanCount + 1); 6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] newspans = new Object[newsize]; 6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] newspanstarts = new int[newsize]; 6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] newspanends = new int[newsize]; 6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] newspanflags = new int[newsize]; 6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mSpans, 0, newspans, 0, mSpanCount); 6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mSpanStarts, 0, newspanstarts, 0, mSpanCount); 6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mSpanEnds, 0, newspanends, 0, mSpanCount); 6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mSpanFlags, 0, newspanflags, 0, mSpanCount); 6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpans = newspans; 6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanStarts = newspanstarts; 6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanEnds = newspanends; 6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanFlags = newspanflags; 6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpans[mSpanCount] = what; 6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanStarts[mSpanCount] = start; 6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanEnds[mSpanCount] = end; 6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanFlags[mSpanCount] = flags; 6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mSpanCount++; 6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 688fc1190b8f143bd00c825d4e52928c577e40e6c49Gilles Debunne if (send) sendSpanAdded(what, nstart, nend); 6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Remove the specified markup object from the buffer. 6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void removeSpan(Object what) { 6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = mSpanCount - 1; i >= 0; i--) { 6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mSpans[i] == what) { 6976435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne removeSpan(i); 6989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return the buffer offset of the beginning of the specified 7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * markup object, or -1 if it is not attached to this buffer. 7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 7079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int getSpanStart(Object what) { 7089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = mSpanCount; 7099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = mSpans; 7109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = count - 1; i >= 0; i--) { 7129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spans[i] == what) { 7139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int where = mSpanStarts[i]; 7149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (where > mGapStart) 7169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project where -= mGapLength; 7179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return where; 7199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return the buffer offset of the end of the specified 7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * markup object, or -1 if it is not attached to this buffer. 7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int getSpanEnd(Object what) { 7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = mSpanCount; 7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = mSpans; 7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = count - 1; i >= 0; i--) { 7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spans[i] == what) { 7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int where = mSpanEnds[i]; 7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (where > mGapStart) 7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project where -= mGapLength; 7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return where; 7419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 7459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 7489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return the flags of the end of the specified 7499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * markup object, or 0 if it is not attached to this buffer. 7509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int getSpanFlags(Object what) { 7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = mSpanCount; 7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = mSpans; 7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = count - 1; i >= 0; i--) { 7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spans[i] == what) { 7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mSpanFlags[i]; 7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return 0; 7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 7659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return an array of the spans of the specified type that overlap 7669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * the specified range of the buffer. The kind may be Object.class to get 7679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * a list of all the spans regardless of type. 7689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 769312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne @SuppressWarnings("unchecked") 7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public <T> T[] getSpans(int queryStart, int queryEnd, Class<T> kind) { 7716435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne if (kind == null) return ArrayUtils.emptyArray(kind); 7726435a56a8c02de98befcc8cd743b2b638cffb327Gilles Debunne 7739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int spanCount = mSpanCount; 7749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = mSpans; 7759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] starts = mSpanStarts; 7769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] ends = mSpanEnds; 7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] flags = mSpanFlags; 7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int gapstart = mGapStart; 7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int gaplen = mGapLength; 7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = 0; 782312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne T[] ret = null; 783312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne T ret1 = null; 7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < spanCount; i++) { 7869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int spanStart = starts[i]; 7879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spanStart > gapstart) { 7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project spanStart -= gaplen; 7899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spanStart > queryEnd) { 7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 793b062e81e3a16af43db3619d721aa522c137d1aa9Gilles Debunne 794b062e81e3a16af43db3619d721aa522c137d1aa9Gilles Debunne int spanEnd = ends[i]; 795b062e81e3a16af43db3619d721aa522c137d1aa9Gilles Debunne if (spanEnd > gapstart) { 796b062e81e3a16af43db3619d721aa522c137d1aa9Gilles Debunne spanEnd -= gaplen; 797b062e81e3a16af43db3619d721aa522c137d1aa9Gilles Debunne } 7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spanEnd < queryStart) { 7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spanStart != spanEnd && queryStart != queryEnd) { 8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spanStart == queryEnd) 8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (spanEnd == queryStart) 8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 809945ee9b1661e60e0074d4f16f61fc147c728c6bfGilles Debunne // Expensive test, should be performed after the previous tests 810945ee9b1661e60e0074d4f16f61fc147c728c6bfGilles Debunne if (!kind.isInstance(spans[i])) continue; 811945ee9b1661e60e0074d4f16f61fc147c728c6bfGilles Debunne 8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count == 0) { 813312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne // Safe conversion thanks to the isInstance test above 814312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne ret1 = (T) spans[i]; 8159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count++; 8169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 8179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count == 1) { 818312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne // Safe conversion, but requires a suppressWarning 819312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne ret = (T[]) Array.newInstance(kind, spanCount - i + 1); 8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret[0] = ret1; 8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int prio = flags[i] & SPAN_PRIORITY; 8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (prio != 0) { 8259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int j; 8269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (j = 0; j < count; j++) { 8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int p = getSpanFlags(ret[j]) & SPAN_PRIORITY; 8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (prio > p) { 8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 8329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(ret, j, ret, j + 1, count - j); 836312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne // Safe conversion thanks to the isInstance test above 837312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne ret[j] = (T) spans[i]; 8389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count++; 8399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 840312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne // Safe conversion thanks to the isInstance test above 841312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne ret[count++] = (T) spans[i]; 8429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count == 0) { 847f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt return ArrayUtils.emptyArray(kind); 8489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count == 1) { 850312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne // Safe conversion, but requires a suppressWarning 851312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne ret = (T[]) Array.newInstance(kind, 1); 8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret[0] = ret1; 853312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne return ret; 8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count == ret.length) { 856312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne return ret; 8579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 859312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne // Safe conversion, but requires a suppressWarning 860312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne T[] nret = (T[]) Array.newInstance(kind, count); 8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(ret, 0, nret, 0, count); 862312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne return nret; 8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 8669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return the next offset after <code>start</code> but less than or 8679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * equal to <code>limit</code> where a span of the specified type 8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * begins or ends. 8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int nextSpanTransition(int start, int limit, Class kind) { 8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = mSpanCount; 8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object[] spans = mSpans; 8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] starts = mSpanStarts; 8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int[] ends = mSpanEnds; 8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int gapstart = mGapStart; 8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int gaplen = mGapLength; 8779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (kind == null) { 8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project kind = Object.class; 8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int st = starts[i]; 8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int en = ends[i]; 8859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (st > gapstart) 8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project st -= gaplen; 8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (en > gapstart) 8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project en -= gaplen; 8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (st > start && st < limit && kind.isInstance(spans[i])) 8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project limit = st; 8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (en > start && en < limit && kind.isInstance(spans[i])) 8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project limit = en; 8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return limit; 8989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return a new CharSequence containing a copy of the specified 9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * range of this buffer, including the overlapping spans. 9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public CharSequence subSequence(int start, int end) { 9059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return new SpannableStringBuilder(this, start, end); 9069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 9099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copy the specified range of chars from this buffer into the 9109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * specified array, beginning at the specified offset. 9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void getChars(int start, int end, char[] dest, int destoff) { 9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project checkRange("getChars", start, end); 9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end <= mGapStart) { 9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mText, start, dest, destoff, end - start); 9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (start >= mGapStart) { 918174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne System.arraycopy(mText, start + mGapLength, dest, destoff, end - start); 9199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mText, start, dest, destoff, mGapStart - start); 9219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.arraycopy(mText, mGapStart + mGapLength, 922174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne dest, destoff + (mGapStart - start), 923174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne end - mGapStart); 9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 9289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Return a String containing a copy of the chars in this buffer. 9299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 930312cd580391d70140eeb4080a4e195039172a6a6Gilles Debunne @Override 9319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public String toString() { 9329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int len = length(); 9339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char[] buf = new char[len]; 9349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project getChars(0, len, buf, 0); 9369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return new String(buf); 9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 939653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne /** 940653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne * Return a String containing a copy of the chars in this buffer, limited to the 941653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne * [start, end[ range. 942653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne * @hide 943653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne */ 944653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne public String substring(int start, int end) { 945653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne char[] buf = new char[end - start]; 946653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne getChars(start, end, buf, 0); 947653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne return new String(buf); 948653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne } 949653d3a27878d5358b4a91518a756f6b9b3407b07Gilles Debunne 950b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne private void sendBeforeTextChanged(TextWatcher[] watchers, int start, int before, int after) { 951b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne int n = watchers.length; 9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < n; i++) { 954b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne watchers[i].beforeTextChanged(this, start, before, after); 9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 958b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne private void sendTextChanged(TextWatcher[] watchers, int start, int before, int after) { 959b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne int n = watchers.length; 9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < n; i++) { 962b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne watchers[i].onTextChanged(this, start, before, after); 9639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 966b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne private void sendAfterTextChanged(TextWatcher[] watchers) { 967b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne int n = watchers.length; 9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < n; i++) { 970b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne watchers[i].afterTextChanged(this); 9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private void sendSpanAdded(Object what, int start, int end) { 9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project SpanWatcher[] recip = getSpans(start, end, SpanWatcher.class); 9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int n = recip.length; 9779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < n; i++) { 9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project recip[i].onSpanAdded(this, what, start, end); 9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private void sendSpanRemoved(Object what, int start, int end) { 9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project SpanWatcher[] recip = getSpans(start, end, SpanWatcher.class); 9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int n = recip.length; 9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 9879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < n; i++) { 9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project recip[i].onSpanRemoved(this, what, start, end); 9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 992174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private void sendSpanChanged(Object what, int oldStart, int oldEnd, int start, int end) { 993174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // The bounds of a possible SpanWatcher are guaranteed to be set before this method is 994174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // called, so that the order of the span does not affect this broadcast. 995174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne SpanWatcher[] spanWatchers = getSpans(Math.min(oldStart, start), 996174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne Math.min(Math.max(oldEnd, end), length()), SpanWatcher.class); 997174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne int n = spanWatchers.length; 9989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < n; i++) { 999174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne spanWatchers[i].onSpanChanged(this, what, oldStart, oldEnd, start, end); 10009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static String region(int start, int end) { 10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return "(" + start + " ... " + end + ")"; 10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private void checkRange(final String operation, int start, int end) { 10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end < start) { 10099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IndexOutOfBoundsException(operation + " " + 1010174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne region(start, end) + " has end before start"); 10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int len = length(); 10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (start > len || end > len) { 10169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IndexOutOfBoundsException(operation + " " + 1017174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne region(start, end) + " ends beyond length " + len); 10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (start < 0 || end < 0) { 10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IndexOutOfBoundsException(operation + " " + 1022174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne region(start, end) + " starts before 0"); 10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1026174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne /* 10279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private boolean isprint(char c) { // XXX 10289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (c >= ' ' && c <= '~') 10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project else 10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int startFlag(int flag) { 10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return (flag >> 4) & 0x0F; 10369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int endFlag(int flag) { 10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return flag & 0x0F; 10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void dump() { // XXX 10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < mGapStart; i++) { 10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print('|'); 10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(' '); 10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(isprint(mText[i]) ? mText[i] : '.'); 10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(' '); 10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = mGapStart; i < mGapStart + mGapLength; i++) { 10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print('|'); 10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print('('); 10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(isprint(mText[i]) ? mText[i] : '.'); 10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(')'); 10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = mGapStart + mGapLength; i < mText.length; i++) { 10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print('|'); 10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(' '); 10609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(isprint(mText[i]) ? mText[i] : '.'); 10619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(' '); 10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print('\n'); 10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < mText.length + 1; i++) { 10679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int found = 0; 10689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int wfound = 0; 10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int j = 0; j < mSpanCount; j++) { 10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mSpanStarts[j] == i) { 10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project found = 1; 10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project wfound = j; 10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mSpanEnds[j] == i) { 10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project found = 2; 10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project wfound = j; 10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (found == 1) { 10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (startFlag(mSpanFlags[wfound]) == MARK) 10869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print("( "); 10879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (startFlag(mSpanFlags[wfound]) == PARAGRAPH) 10889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print("< "); 10899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project else 10909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print("[ "); 10919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (found == 2) { 10929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (endFlag(mSpanFlags[wfound]) == POINT) 10939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(") "); 10949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (endFlag(mSpanFlags[wfound]) == PARAGRAPH) 10959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print("> "); 10969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project else 10979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print("] "); 10989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 10999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print(" "); 11009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project System.out.print("\n"); 11049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 110526b62d432101084c832f5587798cfd24a4ed9c59Gilles Debunne */ 11069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 11089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Don't call this yourself -- exists for Canvas to use internally. 11099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@hide} 11109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1111b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne public void drawText(Canvas c, int start, int end, float x, float y, Paint p) { 11129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project checkRange("drawText", start, end); 11139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end <= mGapStart) { 11159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project c.drawText(mText, start, end - start, x, y, p); 11169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (start >= mGapStart) { 11179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project c.drawText(mText, start + mGapLength, end - start, x, y, p); 11189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 11199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char[] buf = TextUtils.obtain(end - start); 11209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project getChars(start, end, buf, 0); 11229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project c.drawText(buf, 0, end - start, x, y, p); 11239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project TextUtils.recycle(buf); 11249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11270c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 11289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1129f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt * Don't call this yourself -- exists for Canvas to use internally. 1130f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt * {@hide} 1131f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt */ 1132b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne public void drawTextRun(Canvas c, int start, int end, int contextStart, int contextEnd, 1133da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio float x, float y, int flags, Paint p) { 1134f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt checkRange("drawTextRun", start, end); 1135f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt 11360c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt int contextLen = contextEnd - contextStart; 11370c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt int len = end - start; 11380c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt if (contextEnd <= mGapStart) { 1139da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio c.drawTextRun(mText, start, len, contextStart, contextLen, x, y, flags, p); 11400c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } else if (contextStart >= mGapStart) { 11410c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt c.drawTextRun(mText, start + mGapLength, len, contextStart + mGapLength, 1142da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio contextLen, x, y, flags, p); 1143f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt } else { 11440c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt char[] buf = TextUtils.obtain(contextLen); 11450c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt getChars(contextStart, contextEnd, buf, 0); 1146da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio c.drawTextRun(buf, start - contextStart, len, 0, contextLen, x, y, flags, p); 1147f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt TextUtils.recycle(buf); 1148f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt } 1149f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt } 1150f47d7405bbcb25d7cdf89ebb059f41520fe9ab87Doug Felt 1151174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne /** 11529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Don't call this yourself -- exists for Paint to use internally. 11539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@hide} 11549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 11559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public float measureText(int start, int end, Paint p) { 11569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project checkRange("measureText", start, end); 11579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project float ret; 11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end <= mGapStart) { 11619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret = p.measureText(mText, start, end - start); 11629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (start >= mGapStart) { 11639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret = p.measureText(mText, start + mGapLength, end - start); 11649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 11659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char[] buf = TextUtils.obtain(end - start); 11669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project getChars(start, end, buf, 0); 11689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret = p.measureText(buf, 0, end - start); 11699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project TextUtils.recycle(buf); 11709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return ret; 11739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Don't call this yourself -- exists for Paint to use internally. 11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@hide} 11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 11799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int getTextWidths(int start, int end, float[] widths, Paint p) { 11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project checkRange("getTextWidths", start, end); 11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int ret; 11839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (end <= mGapStart) { 11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret = p.getTextWidths(mText, start, end - start, widths); 11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (start >= mGapStart) { 1187174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne ret = p.getTextWidths(mText, start + mGapLength, end - start, widths); 11889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 11899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project char[] buf = TextUtils.obtain(end - start); 11909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project getChars(start, end, buf, 0); 11929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ret = p.getTextWidths(buf, 0, end - start, widths); 11939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project TextUtils.recycle(buf); 11949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return ret; 11979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11990c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt /** 12000c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt * Don't call this yourself -- exists for Paint to use internally. 12010c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt * {@hide} 12020c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt */ 1203da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio public float getTextRunAdvances(int start, int end, int contextStart, int contextEnd, int flags, 12040c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt float[] advances, int advancesPos, Paint p) { 12050c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12060c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt float ret; 12070c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12080c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt int contextLen = contextEnd - contextStart; 12090c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt int len = end - start; 12100c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12110c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt if (end <= mGapStart) { 12120c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt ret = p.getTextRunAdvances(mText, start, len, contextStart, contextLen, 1213da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio flags, advances, advancesPos); 12140c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } else if (start >= mGapStart) { 12150c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt ret = p.getTextRunAdvances(mText, start + mGapLength, len, 1216da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio contextStart + mGapLength, contextLen, flags, advances, advancesPos); 12170c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } else { 12180c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt char[] buf = TextUtils.obtain(contextLen); 12190c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt getChars(contextStart, contextEnd, buf, 0); 12200c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt ret = p.getTextRunAdvances(buf, start - contextStart, len, 1221da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio 0, contextLen, flags, advances, advancesPos); 1222da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio TextUtils.recycle(buf); 1223da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio } 1224da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio 1225da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio return ret; 1226da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio } 1227da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio 1228da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio /** 1229f09d510cd18854f41632433503ca46cc41362febGilles Debunne * Returns the next cursor position in the run. This avoids placing the cursor between 1230f09d510cd18854f41632433503ca46cc41362febGilles Debunne * surrogates, between characters that form conjuncts, between base characters and combining 1231f09d510cd18854f41632433503ca46cc41362febGilles Debunne * marks, or within a reordering cluster. 1232f09d510cd18854f41632433503ca46cc41362febGilles Debunne * 1233f09d510cd18854f41632433503ca46cc41362febGilles Debunne * <p>The context is the shaping context for cursor movement, generally the bounds of the metric 1234f09d510cd18854f41632433503ca46cc41362febGilles Debunne * span enclosing the cursor in the direction of movement. 1235f09d510cd18854f41632433503ca46cc41362febGilles Debunne * <code>contextStart</code>, <code>contextEnd</code> and <code>offset</code> are relative to 1236f09d510cd18854f41632433503ca46cc41362febGilles Debunne * the start of the string.</p> 1237f09d510cd18854f41632433503ca46cc41362febGilles Debunne * 1238616f3835a0a67328b378cb1cbc126f17478ab4a0Gilles Debunne * <p>If cursorOpt is CURSOR_AT and the offset is not a valid cursor position, 1239f09d510cd18854f41632433503ca46cc41362febGilles Debunne * this returns -1. Otherwise this will never return a value before contextStart or after 1240f09d510cd18854f41632433503ca46cc41362febGilles Debunne * contextEnd.</p> 1241f09d510cd18854f41632433503ca46cc41362febGilles Debunne * 1242f09d510cd18854f41632433503ca46cc41362febGilles Debunne * @param contextStart the start index of the context 1243f09d510cd18854f41632433503ca46cc41362febGilles Debunne * @param contextEnd the (non-inclusive) end index of the context 1244da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio * @param flags either DIRECTION_RTL or DIRECTION_LTR 1245f09d510cd18854f41632433503ca46cc41362febGilles Debunne * @param offset the cursor position to move from 1246616f3835a0a67328b378cb1cbc126f17478ab4a0Gilles Debunne * @param cursorOpt how to move the cursor, one of CURSOR_AFTER, 1247616f3835a0a67328b378cb1cbc126f17478ab4a0Gilles Debunne * CURSOR_AT_OR_AFTER, CURSOR_BEFORE, 1248616f3835a0a67328b378cb1cbc126f17478ab4a0Gilles Debunne * CURSOR_AT_OR_BEFORE, or CURSOR_AT 1249f09d510cd18854f41632433503ca46cc41362febGilles Debunne * @param p the Paint object that is requesting this information 1250f09d510cd18854f41632433503ca46cc41362febGilles Debunne * @return the offset of the next position, or -1 1251b0b225602cc84e0602450a417a562e8f53f29c6bGilles Debunne * @deprecated This is an internal method, refrain from using it in your code 1252f09d510cd18854f41632433503ca46cc41362febGilles Debunne */ 1253b0b225602cc84e0602450a417a562e8f53f29c6bGilles Debunne @Deprecated 12540c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt public int getTextRunCursor(int contextStart, int contextEnd, int flags, int offset, 1255da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio int cursorOpt, Paint p) { 12560c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12570c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt int ret; 12580c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12590c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt int contextLen = contextEnd - contextStart; 12600c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt if (contextEnd <= mGapStart) { 12610c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt ret = p.getTextRunCursor(mText, contextStart, contextLen, 1262da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio flags, offset, cursorOpt); 12630c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } else if (contextStart >= mGapStart) { 12640c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt ret = p.getTextRunCursor(mText, contextStart + mGapLength, contextLen, 1265da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio flags, offset + mGapLength, cursorOpt) - mGapLength; 12660c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } else { 12670c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt char[] buf = TextUtils.obtain(contextLen); 12680c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt getChars(contextStart, contextEnd, buf, 0); 12690c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt ret = p.getTextRunCursor(buf, 0, contextLen, 1270da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio flags, offset - contextStart, cursorOpt) + contextStart; 12710c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt TextUtils.recycle(buf); 12720c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } 12730c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12740c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt return ret; 12750c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt } 12760c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt 12779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 12789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void setFilters(InputFilter[] filters) { 12799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (filters == null) { 12809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IllegalArgumentException(); 12819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mFilters = filters; 12849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Documentation from interface 12879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public InputFilter[] getFilters() { 12889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return mFilters; 12899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12919b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase // Same as SpannableStringInternal 12929b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase @Override 12939b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase public boolean equals(Object o) { 12949b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase if (o instanceof Spanned && 12959b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase toString().equals(o.toString())) { 12961d3c4b396a393515e738cb72782ef564362420faChet Haase Spanned other = (Spanned) o; 12979b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase // Check span data 12981d3c4b396a393515e738cb72782ef564362420faChet Haase Object[] otherSpans = other.getSpans(0, other.length(), Object.class); 12999b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase if (mSpanCount == otherSpans.length) { 13009b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase for (int i = 0; i < mSpanCount; ++i) { 13019b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase Object thisSpan = mSpans[i]; 13029b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase Object otherSpan = otherSpans[i]; 13031d3c4b396a393515e738cb72782ef564362420faChet Haase if (thisSpan == this) { 13041d3c4b396a393515e738cb72782ef564362420faChet Haase if (other != otherSpan || 13051d3c4b396a393515e738cb72782ef564362420faChet Haase getSpanStart(thisSpan) != other.getSpanStart(otherSpan) || 13061d3c4b396a393515e738cb72782ef564362420faChet Haase getSpanEnd(thisSpan) != other.getSpanEnd(otherSpan) || 13071d3c4b396a393515e738cb72782ef564362420faChet Haase getSpanFlags(thisSpan) != other.getSpanFlags(otherSpan)) { 13081d3c4b396a393515e738cb72782ef564362420faChet Haase return false; 13091d3c4b396a393515e738cb72782ef564362420faChet Haase } 13101d3c4b396a393515e738cb72782ef564362420faChet Haase } else if (!thisSpan.equals(otherSpan) || 13111d3c4b396a393515e738cb72782ef564362420faChet Haase getSpanStart(thisSpan) != other.getSpanStart(otherSpan) || 13121d3c4b396a393515e738cb72782ef564362420faChet Haase getSpanEnd(thisSpan) != other.getSpanEnd(otherSpan) || 13131d3c4b396a393515e738cb72782ef564362420faChet Haase getSpanFlags(thisSpan) != other.getSpanFlags(otherSpan)) { 13149b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase return false; 13159b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13169b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13179b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase return true; 13189b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13199b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13209b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase return false; 13219b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13229b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase 13239b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase // Same as SpannableStringInternal 13249b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase @Override 13259b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase public int hashCode() { 13269b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase int hash = toString().hashCode(); 13279b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase hash = hash * 31 + mSpanCount; 13289b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase for (int i = 0; i < mSpanCount; ++i) { 13299b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase Object span = mSpans[i]; 13301d3c4b396a393515e738cb72782ef564362420faChet Haase if (span != this) { 13311d3c4b396a393515e738cb72782ef564362420faChet Haase hash = hash * 31 + span.hashCode(); 13321d3c4b396a393515e738cb72782ef564362420faChet Haase } 13339b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase hash = hash * 31 + getSpanStart(span); 13349b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase hash = hash * 31 + getSpanEnd(span); 13359b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase hash = hash * 31 + getSpanFlags(span); 13369b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13379b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase return hash; 13389b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase } 13399b985721dafaa28d1aa34499d23a83afc6d0a2faChet Haase 13409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final InputFilter[] NO_FILTERS = new InputFilter[0]; 13419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private InputFilter[] mFilters = NO_FILTERS; 13429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private char[] mText; 13449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int mGapStart; 13459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int mGapLength; 13469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private Object[] mSpans; 13489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int[] mSpanStarts; 13499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int[] mSpanEnds; 13509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int[] mSpanFlags; 13519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private int mSpanCount; 1352174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private int mSpanCountBeforeAdd; 13539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1354b51036f72aa69e56133fefbf2b56724768423512Gilles Debunne // TODO These value are tightly related to the public SPAN_MARK/POINT values in {@link Spanned} 13550249b43f6ce59bfec104f0fe606d9059244f8797Gilles Debunne private static final int MARK = 1; 13569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int POINT = 2; 13579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int PARAGRAPH = 3; 13589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int START_MASK = 0xF0; 13609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int END_MASK = 0x0F; 13619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int START_SHIFT = 4; 1362174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne 1363174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne // These bits are not (currently) used by SPANNED flags 1364174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private static final int SPAN_START_AT_START = 0x1000; 1365174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private static final int SPAN_START_AT_END = 0x2000; 1366174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private static final int SPAN_END_AT_START = 0x4000; 1367174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private static final int SPAN_END_AT_END = 0x8000; 1368174c44c6cfeead714bb180d21332e712e8b0bbd8Gilles Debunne private static final int SPAN_START_END_MASK = 0xF000; 13699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1370