Files
lawnchair/tests/src/com/android/launcher3/celllayout/ReorderAlgorithmUnitTest.java
Sebastian Franco f7654252b1 Moving ReorderPreviewAnimation into it's own file and rewrite to Kotlin.
This will make it easier to write unit testing.

Fix: 294473336
Test: manual testing
Flag: NA
Change-Id: I2d6cfd8110c5c2ef09c49150a0bd071bc948995c
2024-01-24 14:04:48 -08:00

300 lines
13 KiB
Java

/*
* Copyright (C) 2023 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 com.android.launcher3.celllayout;
import static androidx.test.core.app.ApplicationProvider.getApplicationContext;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;
import android.content.Context;
import android.graphics.Point;
import android.util.Log;
import android.view.View;
import androidx.test.ext.junit.runners.AndroidJUnit4;
import androidx.test.filters.SmallTest;
import com.android.launcher3.CellLayout;
import com.android.launcher3.MultipageCellLayout;
import com.android.launcher3.celllayout.board.CellLayoutBoard;
import com.android.launcher3.celllayout.board.IconPoint;
import com.android.launcher3.celllayout.board.PermutedBoardComparator;
import com.android.launcher3.celllayout.board.WidgetRect;
import com.android.launcher3.celllayout.testgenerator.RandomBoardGenerator;
import com.android.launcher3.celllayout.testgenerator.RandomMultiBoardGenerator;
import com.android.launcher3.util.ActivityContextWrapper;
import com.android.launcher3.views.DoubleShadowBubbleTextView;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.Iterator;
import java.util.List;
import java.util.Random;
@SmallTest
@RunWith(AndroidJUnit4.class)
public class ReorderAlgorithmUnitTest {
private static final String TAG = "ReorderAlgorithmUnitTest";
private static final char MAIN_WIDGET_TYPE = 'z';
// There is nothing special about this numbers, the random seed is just to be able to reproduce
// the test cases and the height and width is a random number similar to what users expect on
// their devices
private static final int SEED = 897;
private static final int MAX_BOARD_SIZE = 13;
private static final int TOTAL_OF_CASES_GENERATED = 300;
private Context mApplicationContext;
@Rule
public UnitTestCellLayoutBuilderRule mCellLayoutBuilder = new UnitTestCellLayoutBuilderRule();
/**
* This test reads existing test cases and makes sure the CellLayout produces the same
* output for each of them for a given input.
*/
@Test
public void testAllCases() throws IOException {
List<ReorderAlgorithmUnitTestCase> testCases = getTestCases(
"ReorderAlgorithmUnitTest/reorder_algorithm_test_cases");
mApplicationContext = new ActivityContextWrapper(getApplicationContext());
List<Integer> failingCases = new ArrayList<>();
for (int i = 0; i < testCases.size(); i++) {
try {
evaluateTestCase(testCases.get(i), false);
} catch (AssertionError e) {
e.printStackTrace();
failingCases.add(i);
}
}
assertEquals("Some test cases failed " + Arrays.toString(failingCases.toArray()), 0,
failingCases.size());
}
/**
* This test generates random CellLayout configurations and then try to reorder it and makes
* sure the result is a valid board meaning it didn't remove any widget or icon.
*/
@Test
public void generateValidTests() {
Random generator = new Random(SEED);
mApplicationContext = new ActivityContextWrapper(getApplicationContext());
for (int i = 0; i < TOTAL_OF_CASES_GENERATED; i++) {
// Using a new seed so that we can replicate the same test cases.
int seed = generator.nextInt();
Log.d(TAG, "Seed = " + seed);
ReorderAlgorithmUnitTestCase testCase = generateRandomTestCase(
new RandomBoardGenerator(new Random(seed))
);
Log.d(TAG, "testCase = " + testCase);
assertTrue("invalid case " + i,
validateIntegrity(testCase.startBoard, testCase.endBoard, testCase));
}
}
/**
* Same as above but testing the Multipage CellLayout.
*/
@Test
public void generateValidTests_Multi() {
Random generator = new Random(SEED);
mApplicationContext = new ActivityContextWrapper(getApplicationContext());
for (int i = 0; i < TOTAL_OF_CASES_GENERATED; i++) {
// Using a new seed so that we can replicate the same test cases.
int seed = generator.nextInt();
Log.d(TAG, "Seed = " + seed);
ReorderAlgorithmUnitTestCase testCase = generateRandomTestCase(
new RandomMultiBoardGenerator(new Random(seed))
);
Log.d(TAG, "testCase = " + testCase);
assertTrue("invalid case " + i,
validateIntegrity(testCase.startBoard, testCase.endBoard, testCase));
}
}
private void addViewInCellLayout(CellLayout cellLayout, int cellX, int cellY, int spanX,
int spanY, boolean isWidget) {
View cell = isWidget ? new View(mApplicationContext) : new DoubleShadowBubbleTextView(
mApplicationContext);
cell.setLayoutParams(new CellLayoutLayoutParams(cellX, cellY, spanX, spanY));
cellLayout.addViewToCellLayout(cell, -1, cell.getId(),
(CellLayoutLayoutParams) cell.getLayoutParams(), true);
}
public ItemConfiguration solve(CellLayoutBoard board, int x, int y, int spanX,
int spanY, int minSpanX, int minSpanY, boolean isMulti) {
CellLayout cl = mCellLayoutBuilder.createCellLayout(board.getWidth(), board.getHeight(),
isMulti);
// The views have to be sorted or the result can vary
board.getIcons()
.stream()
.map(IconPoint::getCoord)
.sorted(Comparator.comparing(p -> ((Point) p).x).thenComparing(p -> ((Point) p).y))
.forEach(p -> addViewInCellLayout(cl, p.x, p.y, 1, 1, false));
board.getWidgets()
.stream()
.sorted(Comparator
.comparing(WidgetRect::getCellX)
.thenComparing(WidgetRect::getCellY)
).forEach(
widget -> addViewInCellLayout(cl, widget.getCellX(), widget.getCellY(),
widget.getSpanX(), widget.getSpanY(), true)
);
int[] testCaseXYinPixels = new int[2];
cl.regionToCenterPoint(x, y, spanX, spanY, testCaseXYinPixels);
ItemConfiguration configuration = new ItemConfiguration();
cl.copyCurrentStateToSolution(configuration);
ItemConfiguration solution = cl.createReorderAlgorithm()
.calculateReorder(
new ReorderParameters(
testCaseXYinPixels[0],
testCaseXYinPixels[1],
spanX,
spanY,
minSpanX,
minSpanY,
null,
configuration
)
);
if (solution == null) {
solution = new ItemConfiguration();
solution.isSolution = false;
}
if (!solution.isSolution) {
cl.copyCurrentStateToSolution(solution);
if (cl instanceof MultipageCellLayout) {
solution =
((MultipageCellLayout) cl).createReorderAlgorithm().removeSeamFromSolution(
solution);
}
solution.isSolution = false;
}
return solution;
}
public CellLayoutBoard boardFromSolution(ItemConfiguration solution, int width,
int height) {
// Update the views with solution value
solution.map.forEach((key, val) -> key.setLayoutParams(
new CellLayoutLayoutParams(val.cellX, val.cellY, val.spanX, val.spanY)));
CellLayoutBoard board = CellLayoutTestUtils.viewsToBoard(
new ArrayList<>(solution.map.keySet()), width, height);
if (solution.isSolution) {
board.addWidget(solution.cellX, solution.cellY, solution.spanX, solution.spanY,
MAIN_WIDGET_TYPE);
}
return board;
}
public void evaluateTestCase(ReorderAlgorithmUnitTestCase testCase, boolean isMultiCellLayout) {
ItemConfiguration solution = solve(testCase.startBoard, testCase.x, testCase.y,
testCase.spanX, testCase.spanY, testCase.minSpanX, testCase.minSpanY,
isMultiCellLayout);
assertEquals("should be a valid solution", solution.isSolution, testCase.isValidSolution);
if (testCase.isValidSolution) {
CellLayoutBoard finishBoard = boardFromSolution(solution,
testCase.startBoard.getWidth(), testCase.startBoard.getHeight());
assertTrue("End result and test case result board doesn't match ",
finishBoard.compareTo(testCase.endBoard) == 0);
}
}
private ReorderAlgorithmUnitTestCase generateRandomTestCase(
RandomBoardGenerator boardGenerator) {
ReorderAlgorithmUnitTestCase testCase = new ReorderAlgorithmUnitTestCase();
boolean isMultiCellLayout = boardGenerator instanceof RandomMultiBoardGenerator;
int width = isMultiCellLayout
? boardGenerator.getRandom(3, MAX_BOARD_SIZE / 2) * 2
: boardGenerator.getRandom(3, MAX_BOARD_SIZE);
int height = boardGenerator.getRandom(3, MAX_BOARD_SIZE);
int targetWidth = boardGenerator.getRandom(1, width - 2);
int targetHeight = boardGenerator.getRandom(1, height - 2);
int minTargetWidth = boardGenerator.getRandom(1, targetWidth);
int minTargetHeight = boardGenerator.getRandom(1, targetHeight);
int x = boardGenerator.getRandom(0, width - targetWidth);
int y = boardGenerator.getRandom(0, height - targetHeight);
CellLayoutBoard board = boardGenerator.generateBoard(width, height,
targetWidth * targetHeight);
ItemConfiguration solution = solve(board, x, y, targetWidth, targetHeight,
minTargetWidth, minTargetHeight, isMultiCellLayout);
CellLayoutBoard finishBoard = boardFromSolution(solution, board.getWidth(),
board.getHeight());
testCase.startBoard = board;
testCase.endBoard = finishBoard;
testCase.isValidSolution = solution.isSolution;
testCase.x = x;
testCase.y = y;
testCase.spanX = targetWidth;
testCase.spanY = targetHeight;
testCase.minSpanX = minTargetWidth;
testCase.minSpanY = minTargetHeight;
testCase.type = solution.area() == 1 ? "icon" : "widget";
return testCase;
}
/**
* Makes sure the final solution has valid integrity meaning that the number and sizes of
* widgets is the expect and there are no missing widgets.
*/
public boolean validateIntegrity(CellLayoutBoard startBoard, CellLayoutBoard finishBoard,
ReorderAlgorithmUnitTestCase testCase) {
if (!testCase.isValidSolution) {
// if we couldn't place the widget then the solution should be identical to the board
return startBoard.compareTo(finishBoard) == 0;
}
WidgetRect addedWidget = finishBoard.getWidgetOfType(MAIN_WIDGET_TYPE);
finishBoard.removeItem(MAIN_WIDGET_TYPE);
Comparator<CellLayoutBoard> comparator = new PermutedBoardComparator();
if (comparator.compare(startBoard, finishBoard) != 0) {
return false;
}
return addedWidget.getSpanX() >= testCase.minSpanX
&& addedWidget.getSpanY() >= testCase.minSpanY;
}
private static List<ReorderAlgorithmUnitTestCase> getTestCases(String testPath)
throws IOException {
List<ReorderAlgorithmUnitTestCase> cases = new ArrayList<>();
Iterator<CellLayoutTestCaseReader.TestSection> iterableSection =
CellLayoutTestCaseReader.readFromFile(testPath).parse().iterator();
while (iterableSection.hasNext()) {
cases.add(ReorderAlgorithmUnitTestCase.readNextCase(iterableSection));
}
return cases;
}
}