| /* |
| * Copyright (C) 2015 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.uirendering.cts.testclasses; |
| |
| import static org.junit.Assert.assertEquals; |
| |
| import android.animation.ValueAnimator; |
| import android.content.pm.PackageManager; |
| import android.graphics.Color; |
| import android.graphics.ColorMatrix; |
| import android.graphics.ColorMatrixColorFilter; |
| import android.graphics.Matrix; |
| import android.graphics.Paint; |
| import android.graphics.Point; |
| import android.graphics.PorterDuff; |
| import android.graphics.PorterDuffXfermode; |
| import android.graphics.Rect; |
| import android.uirendering.cts.R; |
| import android.uirendering.cts.bitmapcomparers.MSSIMComparer; |
| import android.uirendering.cts.bitmapverifiers.ColorCountVerifier; |
| import android.uirendering.cts.bitmapverifiers.ColorVerifier; |
| import android.uirendering.cts.bitmapverifiers.RectVerifier; |
| import android.uirendering.cts.bitmapverifiers.SamplePointVerifier; |
| import android.uirendering.cts.testinfrastructure.ActivityTestBase; |
| import android.uirendering.cts.testinfrastructure.ViewInitializer; |
| import android.uirendering.cts.util.WebViewReadyHelper; |
| import android.view.Gravity; |
| import android.view.View; |
| import android.view.ViewTreeObserver; |
| import android.webkit.WebView; |
| import android.widget.FrameLayout; |
| |
| import androidx.annotation.ColorInt; |
| import androidx.test.filters.LargeTest; |
| import androidx.test.filters.MediumTest; |
| import androidx.test.runner.AndroidJUnit4; |
| |
| import org.junit.Ignore; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| |
| import java.util.concurrent.CountDownLatch; |
| |
| @MediumTest |
| @RunWith(AndroidJUnit4.class) |
| public class LayerTests extends ActivityTestBase { |
| @Test |
| public void testLayerPaintAlpha() { |
| // red channel full strength, other channels 75% strength |
| // (since 25% alpha red subtracts from them) |
| @ColorInt |
| final int expectedColor = Color.rgb(255, 191, 191); |
| createTest() |
| .addLayout(R.layout.simple_red_layout, (ViewInitializer) view -> { |
| // reduce alpha by 50% |
| Paint paint = new Paint(); |
| paint.setAlpha(128); |
| view.setLayerType(View.LAYER_TYPE_HARDWARE, paint); |
| |
| // reduce alpha by another 50% (ensuring two alphas combine correctly) |
| view.setAlpha(0.5f); |
| }) |
| .runWithVerifier(new ColorVerifier(expectedColor)); |
| } |
| |
| @Test |
| public void testLayerPaintSimpleAlphaWithHardware() { |
| @ColorInt |
| final int expectedColor = Color.rgb(255, 128, 128); |
| createTest() |
| .addLayout(R.layout.simple_red_layout, (ViewInitializer) view -> { |
| view.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| |
| // reduce alpha, so that overdraw will result in a different color |
| view.setAlpha(0.5f); |
| }) |
| .runWithVerifier(new ColorVerifier(expectedColor)); |
| } |
| |
| @Test |
| public void testLayerPaintSimpleAlphaWithSoftware() { |
| @ColorInt |
| final int expectedColor = Color.rgb(255, 128, 128); |
| createTest() |
| .addLayout(R.layout.simple_red_layout, (ViewInitializer) view -> { |
| view.setLayerType(View.LAYER_TYPE_SOFTWARE, null); |
| |
| // reduce alpha, so that overdraw will result in a different color |
| view.setAlpha(0.5f); |
| }) |
| .runWithVerifier(new ColorVerifier(expectedColor)); |
| } |
| |
| @Test |
| public void testLayerPaintXfermodeWithSoftware() { |
| createTest() |
| .addLayout(R.layout.simple_red_layout, (ViewInitializer) view -> { |
| Paint paint = new Paint(); |
| paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); |
| view.setLayerType(View.LAYER_TYPE_SOFTWARE, paint); |
| }, true) |
| .runWithVerifier(new ColorVerifier(Color.TRANSPARENT)); |
| } |
| |
| @Test |
| public void testLayerPaintAlphaChanged() { |
| final CountDownLatch fence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.frame_layout, view -> { |
| FrameLayout root = (FrameLayout) view.findViewById(R.id.frame_layout); |
| View child = new View(view.getContext()); |
| child.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| child.setAlpha(0.0f); |
| // add rendering content |
| child.setBackgroundColor(Color.RED); |
| root.addView(child, new FrameLayout.LayoutParams(TEST_WIDTH, TEST_HEIGHT, |
| Gravity.TOP | Gravity.LEFT)); |
| |
| // Post non-zero alpha a few frames in, so that the initial layer draw completes. |
| root.getViewTreeObserver().addOnPreDrawListener( |
| new ViewTreeObserver.OnPreDrawListener() { |
| int mDrawCount = 0; |
| @Override |
| public boolean onPreDraw() { |
| if (mDrawCount++ == 5) { |
| root.getChildAt(0).setAlpha(1.00f); |
| root.getViewTreeObserver().removeOnPreDrawListener(this); |
| root.post(fence::countDown); |
| } else { |
| root.postInvalidate(); |
| } |
| return true; |
| } |
| }); |
| }, true, fence) |
| .runWithVerifier(new ColorVerifier(Color.RED)); |
| } |
| |
| @Test |
| public void testLayerPaintColorFilter() { |
| // Red, fully desaturated. Note that it's not 255/3 in each channel. |
| // See ColorMatrix#setSaturation() |
| @ColorInt |
| final int expectedColor = Color.rgb(54, 54, 54); |
| createTest() |
| .addLayout(R.layout.simple_red_layout, (ViewInitializer) view -> { |
| Paint paint = new Paint(); |
| ColorMatrix desatMatrix = new ColorMatrix(); |
| desatMatrix.setSaturation(0.0f); |
| paint.setColorFilter(new ColorMatrixColorFilter(desatMatrix)); |
| view.setLayerType(View.LAYER_TYPE_HARDWARE, paint); |
| }) |
| .runWithVerifier(new ColorVerifier(expectedColor)); |
| } |
| |
| @Test |
| public void testLayerPaintBlend() { |
| // Red, drawn underneath opaque white, so output should be white. |
| // TODO: consider doing more interesting blending test here |
| @ColorInt |
| final int expectedColor = Color.WHITE; |
| createTest() |
| .addLayout(R.layout.simple_red_layout, (ViewInitializer) view -> { |
| Paint paint = new Paint(); |
| /* Note that when drawing in SW, we're blending within an otherwise empty |
| * SW layer, as opposed to in the frame buffer (which has a white |
| * background). |
| * |
| * For this reason we use just use DST, which just throws out the SRC |
| * content, regardless of the DST alpha channel. |
| */ |
| paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST)); |
| view.setLayerType(View.LAYER_TYPE_HARDWARE, paint); |
| }) |
| .runWithVerifier(new ColorVerifier(expectedColor)); |
| } |
| |
| @LargeTest |
| @Test |
| public void testLayerClear() { |
| ViewInitializer initializer = new ViewInitializer() { |
| ValueAnimator mAnimator; |
| @Override |
| public void initializeView(View view) { |
| FrameLayout root = (FrameLayout) view.findViewById(R.id.frame_layout); |
| root.setAlpha(0.5f); |
| |
| final View child = new View(view.getContext()); |
| child.setBackgroundColor(Color.BLUE); |
| child.setTranslationX(10); |
| child.setLayoutParams( |
| new FrameLayout.LayoutParams(50, 50)); |
| child.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| root.addView(child); |
| |
| mAnimator = ValueAnimator.ofInt(0, 20); |
| mAnimator.setRepeatMode(ValueAnimator.REVERSE); |
| mAnimator.setRepeatCount(ValueAnimator.INFINITE); |
| mAnimator.setDuration(200); |
| mAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { |
| @Override |
| public void onAnimationUpdate(ValueAnimator animation) { |
| child.setTranslationY((Integer) mAnimator.getAnimatedValue()); |
| } |
| }); |
| mAnimator.start(); |
| } |
| @Override |
| public void teardownView() { |
| mAnimator.cancel(); |
| } |
| }; |
| |
| createTest() |
| .addLayout(R.layout.frame_layout, initializer, true) |
| .runWithAnimationVerifier(new ColorCountVerifier( |
| Color.WHITE, 90 * 90 - 50 * 50, 10)); |
| } |
| |
| @Test |
| public void testAlphaLayerChild() { |
| ViewInitializer initializer = new ViewInitializer() { |
| @Override |
| public void initializeView(View view) { |
| FrameLayout root = (FrameLayout) view.findViewById(R.id.frame_layout); |
| root.setAlpha(0.5f); |
| |
| View child = new View(view.getContext()); |
| child.setBackgroundColor(Color.BLUE); |
| child.setTranslationX(10); |
| child.setTranslationY(10); |
| child.setLayoutParams( |
| new FrameLayout.LayoutParams(50, 50)); |
| child.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| root.addView(child); |
| } |
| }; |
| |
| createTest() |
| .addLayout(R.layout.frame_layout, initializer) |
| .runWithVerifier(new RectVerifier(Color.WHITE, 0xff8080ff, |
| new Rect(10, 10, 60, 60))); |
| } |
| |
| @Test |
| public void testLayerInitialSizeZero() { |
| createTest() |
| .addLayout(R.layout.frame_layout, view -> { |
| FrameLayout root = (FrameLayout) view.findViewById(R.id.frame_layout); |
| // disable clipChildren, to ensure children aren't rejected by bounds |
| root.setClipChildren(false); |
| for (int i = 0; i < 2; i++) { |
| View child = new View(view.getContext()); |
| child.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| // add rendering content, so View isn't skipped at render time |
| child.setBackgroundColor(Color.RED); |
| |
| // add one with width=0, one with height=0 |
| root.addView(child, new FrameLayout.LayoutParams( |
| i == 0 ? 0 : 90, |
| i == 0 ? 90 : 0, |
| Gravity.TOP | Gravity.LEFT)); |
| } |
| }, true) |
| .runWithVerifier(new ColorVerifier(Color.WHITE, 0 /* zero tolerance */)); |
| } |
| |
| @Test |
| public void testLayerResizeZero() { |
| final CountDownLatch fence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.frame_layout, view -> { |
| FrameLayout root = (FrameLayout) view.findViewById(R.id.frame_layout); |
| // disable clipChildren, to ensure child isn't rejected by bounds |
| root.setClipChildren(false); |
| for (int i = 0; i < 2; i++) { |
| View child = new View(view.getContext()); |
| child.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| // add rendering content, so View isn't skipped at render time |
| child.setBackgroundColor(Color.BLUE); |
| root.addView(child, new FrameLayout.LayoutParams(90, 90, |
| Gravity.TOP | Gravity.LEFT)); |
| } |
| |
| // post invalid dimensions a few frames in, so initial layer allocation succeeds |
| // NOTE: this must execute before capture, or verification will fail |
| root.getViewTreeObserver().addOnPreDrawListener( |
| new ViewTreeObserver.OnPreDrawListener() { |
| int mDrawCount = 0; |
| @Override |
| public boolean onPreDraw() { |
| if (mDrawCount++ == 5) { |
| root.getChildAt(0).getLayoutParams().width = 0; |
| root.getChildAt(0).requestLayout(); |
| root.getChildAt(1).getLayoutParams().height = 0; |
| root.getChildAt(1).requestLayout(); |
| root.getViewTreeObserver().removeOnPreDrawListener(this); |
| root.post(fence::countDown); |
| } else { |
| root.postInvalidate(); |
| } |
| return true; |
| } |
| }); |
| }, true, fence) |
| .runWithVerifier(new ColorVerifier(Color.WHITE, 0 /* zero tolerance */)); |
| } |
| |
| @Test |
| public void testSaveLayerWithColorFilter() { |
| // verify that renderer can draw nested clipped layers with chained color filters |
| createTest() |
| .addCanvasClient((canvas, width, height) -> { |
| Paint redPaint = new Paint(); |
| redPaint.setColor(0xffff0000); |
| Paint firstLayerPaint = new Paint(); |
| float[] blueToGreenMatrix = new float[20]; |
| blueToGreenMatrix[7] = blueToGreenMatrix[18] = 1.0f; |
| ColorMatrixColorFilter blueToGreenFilter = new ColorMatrixColorFilter(blueToGreenMatrix); |
| firstLayerPaint.setColorFilter(blueToGreenFilter); |
| Paint secondLayerPaint = new Paint(); |
| float[] redToBlueMatrix = new float[20]; |
| redToBlueMatrix[10] = redToBlueMatrix[18] = 1.0f; |
| ColorMatrixColorFilter redToBlueFilter = new ColorMatrixColorFilter(redToBlueMatrix); |
| secondLayerPaint.setColorFilter(redToBlueFilter); |
| // The color filters are applied starting first with the inner layer and then the |
| // outer layer. |
| canvas.saveLayer(40, 5, 80, 70, firstLayerPaint); |
| canvas.saveLayer(5, 40, 70, 80, secondLayerPaint); |
| canvas.drawRect(10, 10, 70, 70, redPaint); |
| canvas.restore(); |
| canvas.restore(); |
| }) |
| .runWithVerifier(new RectVerifier(Color.WHITE, Color.GREEN, new Rect(40, 40, 70, 70))); |
| } |
| |
| @Test |
| public void testSaveLayerWithAlpha() { |
| // verify that renderer can draw nested clipped layers with different alpha |
| createTest() // picture mode is disable due to bug:34871089 |
| .addCanvasClient((canvas, width, height) -> { |
| Paint redPaint = new Paint(); |
| redPaint.setColor(0xffff0000); |
| canvas.saveLayerAlpha(40, 5, 80, 70, 0x7f); |
| canvas.saveLayerAlpha(5, 40, 70, 80, 0x3f); |
| canvas.drawRect(10, 10, 70, 70, redPaint); |
| canvas.restore(); |
| canvas.restore(); |
| }) |
| .runWithVerifier(new RectVerifier(Color.WHITE, 0xffffE0E0, new Rect(40, 40, 70, 70))); |
| } |
| |
| @Test |
| public void testSaveLayerRestoreBehavior() { |
| createTest() |
| .addCanvasClient((canvas, width, height) -> { |
| //set identity matrix |
| Matrix identity = new Matrix(); |
| canvas.setMatrix(identity); |
| final Paint p = new Paint(); |
| |
| canvas.saveLayer(0, 0, width, height, p); |
| |
| //change matrix and clip to something different |
| canvas.clipRect(0, 0, width >> 1, height >> 1); |
| Matrix scaledMatrix = new Matrix(); |
| scaledMatrix.setScale(4, 5); |
| canvas.setMatrix(scaledMatrix); |
| assertEquals(scaledMatrix, canvas.getMatrix()); |
| |
| canvas.drawColor(Color.BLUE); |
| canvas.restore(); |
| |
| //check if identity matrix is restored |
| assertEquals(identity, canvas.getMatrix()); |
| |
| //should draw to the entire canvas, because clip has been removed |
| canvas.drawColor(Color.RED); |
| }) |
| .runWithVerifier(new ColorVerifier(Color.RED)); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewWithLayer() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.test_content_webview, (ViewInitializer) view -> { |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| webview.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| }, true, hwFence) |
| .runWithVerifier(new ColorVerifier(Color.BLUE)); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewWithOffsetLayer() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.frame_layout_webview, (ViewInitializer) view -> { |
| FrameLayout layout = view.requireViewById(R.id.frame_layout); |
| layout.setBackgroundColor(Color.RED); |
| |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| |
| webview.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| webview.setTranslationX(10); |
| webview.setTranslationY(10); |
| webview.getLayoutParams().width = TEST_WIDTH - 20; |
| webview.getLayoutParams().height = TEST_HEIGHT - 20; |
| }, true, hwFence) |
| .runWithVerifier(new RectVerifier(Color.RED, Color.BLUE, |
| new Rect(10, 10, TEST_WIDTH - 10, TEST_HEIGHT - 10))); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewWithParentLayer() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.frame_layout_webview, (ViewInitializer) view -> { |
| FrameLayout layout = view.requireViewById(R.id.frame_layout); |
| layout.setBackgroundColor(Color.RED); |
| layout.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| |
| webview.setTranslationX(10); |
| webview.setTranslationY(10); |
| webview.getLayoutParams().width = TEST_WIDTH - 20; |
| webview.getLayoutParams().height = TEST_HEIGHT - 20; |
| |
| }, true, hwFence) |
| .runWithVerifier(new RectVerifier(Color.RED, Color.BLUE, |
| new Rect(10, 10, TEST_WIDTH - 10, TEST_HEIGHT - 10))); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewScaledWithParentLayer() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.frame_layout_webview, (ViewInitializer) view -> { |
| FrameLayout layout = view.requireViewById(R.id.frame_layout); |
| layout.setBackgroundColor(Color.RED); |
| layout.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| |
| webview.setTranslationX(10); |
| webview.setTranslationY(10); |
| webview.setScaleX(0.5f); |
| webview.getLayoutParams().width = 40; |
| webview.getLayoutParams().height = 40; |
| |
| }, true, hwFence) |
| .runWithVerifier(new RectVerifier(Color.RED, Color.BLUE, |
| new Rect(20, 10, 40, 50))); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewWithAlpha() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.test_content_webview, (ViewInitializer) view -> { |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| |
| // reduce alpha by 50% |
| webview.setAlpha(0.5f); |
| |
| }, true, hwFence) |
| .runWithVerifier(new ColorVerifier(Color.rgb(128, 128, 255))); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewWithAlphaLayer() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.test_content_webview, (ViewInitializer) view -> { |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| |
| // reduce alpha by 50% |
| Paint paint = new Paint(); |
| paint.setAlpha(128); |
| webview.setLayerType(View.LAYER_TYPE_HARDWARE, paint); |
| |
| // reduce alpha by another 50% (ensuring two alphas combine correctly) |
| webview.setAlpha(0.5f); |
| |
| }, true, hwFence) |
| .runWithVerifier(new ColorVerifier(Color.rgb(191, 191, 255))); |
| } |
| |
| @LargeTest |
| @Test |
| @Ignore // b/109839751 |
| public void testWebViewWithUnclippedLayer() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.test_content_webview, (ViewInitializer) view -> { |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"min-height: 120vh; background-color:blue\">"); |
| webview.setVerticalFadingEdgeEnabled(true); |
| webview.setVerticalScrollBarEnabled(false); |
| webview.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| |
| }, true, hwFence) |
| .runWithVerifier(new SamplePointVerifier( |
| new Point[] { |
| // solid area |
| new Point(0, 0), |
| new Point(0, TEST_HEIGHT - 1), |
| // fade area |
| new Point(0, TEST_HEIGHT - 10), |
| new Point(0, TEST_HEIGHT - 5) |
| }, |
| new int[] { |
| Color.BLUE, |
| Color.WHITE, |
| 0xffb3b3ff, // white blended with blue |
| 0xffdbdbff // white blended with blue |
| })); |
| } |
| |
| @LargeTest |
| @Test |
| @Ignore // b/109839751 |
| public void testWebViewWithUnclippedLayerAndComplexClip() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| .addLayout(R.layout.circle_clipped_webview, (ViewInitializer) view -> { |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"min-height: 120vh; background-color:blue\">"); |
| webview.setVerticalFadingEdgeEnabled(true); |
| webview.setVerticalScrollBarEnabled(false); |
| webview.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| |
| }, true, hwFence) |
| .runWithVerifier(new SamplePointVerifier( |
| new Point[] { |
| // solid white area |
| new Point(0, 0), |
| new Point(0, TEST_HEIGHT - 1), |
| // solid blue area |
| new Point(TEST_WIDTH / 2 , 5), |
| // fade area |
| new Point(TEST_WIDTH / 2, TEST_HEIGHT - 10), |
| new Point(TEST_WIDTH / 2, TEST_HEIGHT - 5) |
| }, |
| new int[] { |
| Color.WHITE, |
| Color.WHITE, |
| Color.BLUE, |
| 0xffb3b3ff, // white blended with blue |
| 0xffdbdbff // white blended with blue |
| })); |
| } |
| |
| @LargeTest |
| @Test |
| public void testWebViewWithLayerAndComplexClip() { |
| if (!getActivity().getPackageManager().hasSystemFeature(PackageManager.FEATURE_WEBVIEW)) { |
| return; // no WebView to run test on |
| } |
| CountDownLatch hwFence = new CountDownLatch(1); |
| createTest() |
| // golden client - draw a simple non-AA circle |
| .addCanvasClient((canvas, width, height) -> { |
| Paint paint = new Paint(); |
| paint.setAntiAlias(false); |
| paint.setColor(Color.BLUE); |
| canvas.drawOval(0, 0, width, height, paint); |
| }, false) |
| // verify against solid color webview, clipped to its parent oval |
| .addLayout(R.layout.circle_clipped_webview, (ViewInitializer) view -> { |
| FrameLayout layout = view.requireViewById(R.id.circle_clip_frame_layout); |
| layout.setLayerType(View.LAYER_TYPE_HARDWARE, null); |
| |
| WebView webview = view.requireViewById(R.id.webview); |
| WebViewReadyHelper helper = new WebViewReadyHelper(webview, hwFence); |
| helper.loadData("<body style=\"background-color:blue\">"); |
| |
| }, true, hwFence) |
| .runWithComparer(new MSSIMComparer(0.95)); |
| } |
| } |