diff options
Diffstat (limited to 'core')
-rw-r--r-- | core/java/android/text/DynamicLayout.java | 18 | ||||
-rw-r--r-- | core/tests/coretests/src/android/text/DynamicLayoutBlocksTest.java | 327 |
2 files changed, 344 insertions, 1 deletions
diff --git a/core/java/android/text/DynamicLayout.java b/core/java/android/text/DynamicLayout.java index 5f2d642..ff5a467 100644 --- a/core/java/android/text/DynamicLayout.java +++ b/core/java/android/text/DynamicLayout.java @@ -378,12 +378,16 @@ public class DynamicLayout extends Layout * An index is associated to each block (which will be used by display lists), * this class simply invalidates the index of blocks overlapping a modification. * + * This method is package private and not private so that it can be tested. + * * @param startLine the first line of the range of modified lines * @param endLine the last line of the range, possibly equal to startLine, lower * than getLineCount() * @param newLineCount the number of lines that will replace the range, possibly 0 + * + * @hide */ - private void updateBlocks(int startLine, int endLine, int newLineCount) { + void updateBlocks(int startLine, int endLine, int newLineCount) { int firstBlock = -1; int lastBlock = -1; for (int i = 0; i < mNumberOfBlocks; i++) { @@ -466,6 +470,18 @@ public class DynamicLayout extends Layout } /** + * This package private method is used for test purposes only + * @hide + */ + void setBlocksDataForTest(int[] blockEnds, int[] blockIndices, int numberOfBlocks) { + mBlockEnds = new int[blockEnds.length]; + mBlockIndices = new int[blockIndices.length]; + System.arraycopy(blockEnds, 0, mBlockEnds, 0, blockEnds.length); + System.arraycopy(blockIndices, 0, mBlockIndices, 0, blockIndices.length); + mNumberOfBlocks = numberOfBlocks; + } + + /** * @hide */ public int[] getBlockEnds() { diff --git a/core/tests/coretests/src/android/text/DynamicLayoutBlocksTest.java b/core/tests/coretests/src/android/text/DynamicLayoutBlocksTest.java new file mode 100644 index 0000000..73da84f --- /dev/null +++ b/core/tests/coretests/src/android/text/DynamicLayoutBlocksTest.java @@ -0,0 +1,327 @@ +/* + * Copyright (C) 2012 The Android Open Source Project + * + * Licensed under the Apache License, Version 2.0 (the "License"); you may not + * use this file except in compliance with the License. You may obtain a copy of + * the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations under + * the License. + */ + +package android.text; + +import static android.text.Layout.Alignment.*; + +import android.text.DynamicLayout; + +import junit.framework.TestCase; + +/** + * Tests DynamciLayout updateBlocks method. + * + * Requires disabling access checks in the vm since this calls package-private APIs. + * + * @Suppress + */ +public class DynamicLayoutBlocksTest extends TestCase { + private DynamicLayout dl = new DynamicLayout("", new TextPaint(), 0, ALIGN_NORMAL, 0, 0, false); + private static final int ___ = DynamicLayout.INVALID_BLOCK_INDEX; + + private int[] initialBlockEnds; + private int[] initialBlockIndices; + + private void defineInitialState(int[] ends, int[] indices) { + initialBlockEnds = ends; + initialBlockIndices = indices; + assertEquals(initialBlockEnds.length, initialBlockIndices.length); + } + + public void printBlocks(String message) { + System.out.print(message); + for (int i = 0; i < dl.getNumberOfBlocks(); i++) { + System.out.print(" " + Integer.toString(dl.getBlockEnds()[i])); + } + System.out.println(); + } + + public void checkInvariants() { + assertTrue(dl.getNumberOfBlocks() > 0); + assertTrue(dl.getNumberOfBlocks() <= dl.getBlockEnds().length); + assertEquals(dl.getBlockEnds().length, dl.getBlockIndices().length); + + for (int i = 1; i < dl.getNumberOfBlocks(); i++) { + assertTrue(dl.getBlockEnds()[i] > dl.getBlockEnds()[i-1]); + } + } + + private void update(int startLine, int endLine, int newLineCount) { + dl.setBlocksDataForTest(initialBlockEnds, initialBlockIndices, initialBlockEnds.length); + checkInvariants(); + dl.updateBlocks(startLine, endLine, newLineCount); + } + + private void assertState(int[] sizes, int[] indices) { + checkInvariants(); + + assertEquals(sizes.length, dl.getNumberOfBlocks()); + assertEquals(indices.length, dl.getNumberOfBlocks()); + + int[] ends = new int[sizes.length]; + for (int i = 0; i < ends.length; i++) { + ends[i] = i == 0 ? (sizes[0] == 0 ? 0 : sizes[0] - 1) : ends[i - 1] + sizes[i]; + } + + for (int i = 0; i < dl.getNumberOfBlocks(); i++) { + assertEquals(ends[i], dl.getBlockEnds()[i]); + assertEquals(indices[i], dl.getBlockIndices()[i]); + } + } + + private void assertState(int[] sizes) { + int[] ids = new int[sizes.length]; + for (int i = 0; i < sizes.length; i++) { + ids[i] = DynamicLayout.INVALID_BLOCK_INDEX; + } + assertState(sizes, ids); + } + + public void testFrom0() { + defineInitialState( new int[] { 0 }, new int[] { 123 }); + + update(0, 0, 0); + assertState( new int[] { 0 } ); + + update(0, 0, 1); + assertState( new int[] { 0 } ); + + update(0, 0, 10); + assertState( new int[] { 10 } ); + } + + public void testFrom1ReplaceByEmpty() { + defineInitialState( new int[] { 100 }, new int[] { 123 }); + + update(0, 0, 0); + assertState( new int[] { 100 } ); + + update(0, 10, 0); + assertState( new int[] { 90 } ); + + update(0, 100, 0); + assertState( new int[] { 0 } ); + + update(20, 30, 0); + assertState( new int[] { 20, 70 } ); + + update(20, 20, 0); + assertState( new int[] { 20, 80 } ); + + update(40, 100, 0); + assertState( new int[] { 40 } ); + + update(100, 100, 0); + assertState( new int[] { 100 } ); + } + + public void testFrom1ReplaceFromFirstLine() { + defineInitialState( new int[] { 100 }, new int[] { 123 }); + + update(0, 0, 1); + assertState( new int[] { 0, 100 } ); + + update(0, 0, 10); + assertState( new int[] { 10, 100 } ); + + update(0, 30, 31); + assertState( new int[] { 31, 70 } ); + + update(0, 100, 20); + assertState( new int[] { 20 } ); + } + + public void testFrom1ReplaceFromCenter() { + defineInitialState( new int[] { 100 }, new int[] { 123 }); + + update(20, 20, 1); + assertState( new int[] { 20, 1, 80 } ); + + update(20, 20, 10); + assertState( new int[] { 20, 10, 80 } ); + + update(20, 30, 50); + assertState( new int[] { 20, 50, 70 } ); + + update(20, 100, 50); + assertState( new int[] { 20, 50 } ); + } + + public void testFrom1ReplaceFromEnd() { + defineInitialState( new int[] { 100 }, new int[] { 123 }); + + update(100, 100, 0); + assertState( new int[] { 100 } ); + + update(100, 100, 1); + assertState( new int[] { 100, 1 } ); + + update(100, 100, 10); + assertState( new int[] { 100, 10 } ); + } + + public void testFrom2ReplaceFromFirstLine() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(0, 4, 50); + assertState( new int[] { 50, 10-4, 20-10 }, new int[] { ___, ___, 456 } ); + + update(0, 10, 50); + assertState( new int[] { 50, 20-10 }, new int[] { ___, 456 } ); + + update(0, 15, 50); + assertState( new int[] { 50, 20-15 }, new int[] { ___, ___ } ); + + update(0, 20, 50); + assertState( new int[] { 50 }, new int[] { ___ } ); + } + + public void testFrom2ReplaceFromFirstBlock() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(3, 7, 50); + assertState( new int[] { 3, 50, 10-7, 20-10 }, new int[] { ___, ___, ___, 456 } ); + + update(3, 10, 50); + assertState( new int[] { 3, 50, 20-10 }, new int[] { ___, ___, 456 } ); + + update(3, 14, 50); + assertState( new int[] { 3, 50, 20-14 }, new int[] { ___, ___, ___ } ); + + update(3, 20, 50); + assertState( new int[] { 3, 50 }, new int[] { ___, ___ } ); + } + + public void testFrom2ReplaceFromBottomBoundary() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(10, 10, 50); + assertState( new int[] { 10, 50, 20-10 }, new int[] { ___, ___, 456 } ); + + update(10, 14, 50); + assertState( new int[] { 10, 50, 20-14 }, new int[] { ___, ___, ___ } ); + + update(10, 20, 50); + assertState( new int[] { 10, 50 }, new int[] { ___, ___ } ); + } + + public void testFrom2ReplaceFromTopBoundary() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(11, 11, 50); + assertState( new int[] { 11, 50, 20-11 }, new int[] { 123, ___, ___ } ); + + update(11, 14, 50); + assertState( new int[] { 11, 50, 20-14 }, new int[] { 123, ___, ___ } ); + + update(11, 20, 50); + assertState( new int[] { 11, 50 }, new int[] { 123, ___ } ); + } + + public void testFrom2ReplaceFromSecondBlock() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(14, 14, 50); + assertState( new int[] { 11, 14-11, 50, 20-14 }, new int[] { 123, ___, ___, ___ } ); + + update(14, 17, 50); + assertState( new int[] { 11, 14-11, 50, 20-17 }, new int[] { 123, ___, ___, ___ } ); + + update(14, 20, 50); + assertState( new int[] { 11, 14-11, 50 }, new int[] { 123, ___, ___ } ); + } + + public void testFrom2RemoveFromFirst() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(0, 4, 0); + assertState( new int[] { 10-4, 20-10 }, new int[] { ___, 456 } ); + + update(0, 10, 0); + assertState( new int[] { 20-10 }, new int[] { 456 } ); + + update(0, 14, 0); + assertState( new int[] { 20-14 }, new int[] { ___ } ); + + update(0, 20, 0); + assertState( new int[] { 0 }, new int[] { ___ } ); + } + + public void testFrom2RemoveFromFirstBlock() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(4, 7, 0); + assertState( new int[] { 4, 10-7, 20-10 }, new int[] { ___, ___, 456 } ); + + update(4, 10, 0); + assertState( new int[] { 4, 20-10 }, new int[] { ___, 456 } ); + + update(4, 14, 0); + assertState( new int[] { 4, 20-14 }, new int[] { ___, ___ } ); + + update(4, 20, 0); + assertState( new int[] { 4 }, new int[] { ___ } ); + } + + public void testFrom2RemoveFromSecondBlock() { + defineInitialState( new int[] { 10, 20 }, new int[] { 123, 456 }); + + update(14, 17, 0); + assertState( new int[] { 11, 14-11, 20-17 }, new int[] { 123, ___, ___ } ); + + update(14, 20, 0); + assertState( new int[] { 11, 14-11 }, new int[] { 123, ___ } ); + } + + public void testFrom3ReplaceFromFirstBlock() { + defineInitialState( new int[] { 10, 30, 60 }, new int[] { 123, 456, 789 }); + + update(3, 7, 50); + assertState( new int[] { 3, 50, 10-7, 30-10, 60-30 }, new int[] { ___, ___, ___, 456, 789 } ); + + update(3, 10, 50); + assertState( new int[] { 3, 50, 30-10, 60-30 }, new int[] { ___, ___, 456, 789 } ); + + update(3, 17, 50); + assertState( new int[] { 3, 50, 30-17, 60-30 }, new int[] { ___, ___, ___, 789 } ); + + update(3, 30, 50); + assertState( new int[] { 3, 50, 60-30 }, new int[] { ___, ___, 789 } ); + + update(3, 40, 50); + assertState( new int[] { 3, 50, 60-40 }, new int[] { ___, ___, ___ } ); + + update(3, 60, 50); + assertState( new int[] { 3, 50 }, new int[] { ___, ___ } ); + } + + public void testFrom3ReplaceFromSecondBlock() { + defineInitialState( new int[] { 10, 30, 60 }, new int[] { 123, 456, 789 }); + + update(13, 17, 50); + assertState( new int[] { 11, 2, 50, 30-17, 60-30 }, new int[] { 123, ___, ___, ___, 789 } ); + + update(13, 30, 50); + assertState( new int[] { 11, 2, 50, 60-30 }, new int[] { 123, ___, ___, 789 } ); + + update(13, 40, 50); + assertState( new int[] { 11, 2, 50, 60-40 }, new int[] { 123, ___, ___, ___ } ); + + update(13, 60, 50); + assertState( new int[] { 11, 2, 50 }, new int[] { 123, ___, ___ } ); + } +} |