Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 810a8676 authored by Chet Haase's avatar Chet Haase
Browse files

Corrects invalidation logic for layered views

A bug in the invalidation logic meant that changes to a view
would not cause parents in the view hiearchy that were set to have
a layer (e.g., View.LAYER_TYPE_HARDWARE) to get invalidated properly.
So even though the child view was all set to recreate its display list
according to the property change, the layer in the tree above it would stay
as-is, meaning that the change would not show up on the screen.

Issue #5887530 DropTarget text does not change color with the icon

Change-Id: Ie6eac4f406d172cb437822d9fe76340ab2afaf1c
parent 73b61d67
Loading
Loading
Loading
Loading
+2 −59
Original line number Diff line number Diff line
@@ -3938,59 +3938,6 @@ public abstract class ViewGroup extends View implements ViewParent, ViewManager
            // through
            final boolean drawAnimation = (child.mPrivateFlags & DRAW_ANIMATION) == DRAW_ANIMATION;

            //noinspection PointlessBooleanExpression
            if (!HardwareRenderer.RENDER_DIRTY_REGIONS) {
                if (dirty == null) {
                    if (child.mLayerType != LAYER_TYPE_NONE) {
                        mPrivateFlags |= INVALIDATED;
                        mPrivateFlags &= ~DRAWING_CACHE_VALID;
                        child.mLocalDirtyRect.setEmpty();
                    }
                    do {
                        View view = null;
                        if (parent instanceof View) {
                            view = (View) parent;
                            if (view.mLayerType != LAYER_TYPE_NONE) {
                                view.mLocalDirtyRect.setEmpty();
                                if (view.getParent() instanceof View) {
                                    final View grandParent = (View) view.getParent();
                                    grandParent.mPrivateFlags |= INVALIDATED;
                                    grandParent.mPrivateFlags &= ~DRAWING_CACHE_VALID;
                                }
                            }
                            if ((view.mPrivateFlags & DIRTY_MASK) != 0) {
                                // already marked dirty - we're done
                                break;
                            }
                        }
    
                        if (drawAnimation) {
                            if (view != null) {
                                view.mPrivateFlags |= DRAW_ANIMATION;
                            } else if (parent instanceof ViewRootImpl) {
                                ((ViewRootImpl) parent).mIsAnimating = true;
                            }
                        }
    
                        if (parent instanceof ViewRootImpl) {
                            ((ViewRootImpl) parent).invalidate();
                            parent = null;
                        } else if (view != null) {
                            if ((view.mPrivateFlags & DRAWN) == DRAWN ||
                                    (view.mPrivateFlags & DRAWING_CACHE_VALID) == DRAWING_CACHE_VALID) {
                                view.mPrivateFlags &= ~DRAWING_CACHE_VALID;
                                view.mPrivateFlags |= DIRTY;
                                parent = view.mParent;
                            } else {
                                parent = null;
                            }
                        }
                    } while (parent != null);
                }

                return;
            }

            // Check whether the child that requests the invalidate is fully opaque
            // Views being animated or transformed are not considered opaque because we may
            // be invalidating their old position and need the parent to paint behind them.
@@ -4025,12 +3972,6 @@ public abstract class ViewGroup extends View implements ViewParent, ViewManager
                View view = null;
                if (parent instanceof View) {
                    view = (View) parent;
                    if (view.mLayerType != LAYER_TYPE_NONE &&
                            view.getParent() instanceof View) {
                        final View grandParent = (View) view.getParent();
                        grandParent.mPrivateFlags |= INVALIDATED;
                        grandParent.mPrivateFlags &= ~DRAWING_CACHE_VALID;
                    }
                }

                if (drawAnimation) {
@@ -4103,6 +4044,7 @@ public abstract class ViewGroup extends View implements ViewParent, ViewManager
                    location[CHILD_TOP_INDEX] = top;

                    if (mLayerType != LAYER_TYPE_NONE) {
                        mPrivateFlags |= INVALIDATED;
                        mLocalDirtyRect.union(dirty);
                    }

@@ -4121,6 +4063,7 @@ public abstract class ViewGroup extends View implements ViewParent, ViewManager
                }

                if (mLayerType != LAYER_TYPE_NONE) {
                    mPrivateFlags |= INVALIDATED;
                    mLocalDirtyRect.union(dirty);
                }

+9 −0
Original line number Diff line number Diff line
@@ -675,5 +675,14 @@
            </intent-filter>
        </activity>

        <activity
                android:name="ViewLayerInvalidationActivity"
                android:label="_ViewLayerInvalidation">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

    </application>
</manifest>
+98 −0
Original line number Diff line number Diff line
<?xml version="1.0" encoding="utf-8"?>
<!--
**
** Copyright 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.
*/
-->

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="fill_parent"
              android:layout_height="fill_parent">
    <LinearLayout android:orientation="horizontal"
                  android:layout_width="wrap_content"
                  android:layout_height="wrap_content">
        <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="This is some text"
                android:id="@+id/nestedStatus"/>
        <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="This is some text"
                android:id="@+id/invalidateStatus"/>
    </LinearLayout>
    <LinearLayout android:orientation="vertical"
                  android:layout_width="fill_parent"
                  android:layout_height="fill_parent"
                  android:id="@+id/container">
        <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="This is some text"
                android:id="@+id/textview"/>
        <TextView
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="This is some text"
                android:id="@+id/textviewa"/>
        <LinearLayout android:orientation="vertical"
                      android:layout_width="wrap_content"
                      android:layout_height="wrap_content"
                      android:id="@+id/container1">
            <TextView
                    android:layout_width="wrap_content"
                    android:layout_height="wrap_content"
                    android:text="This is some text"
                    android:id="@+id/textview1"/>
        </LinearLayout>
        <LinearLayout android:orientation="vertical"
                      android:layout_width="wrap_content"
                      android:layout_height="wrap_content"
                      android:id="@+id/container2">
            <LinearLayout android:orientation="vertical"
                          android:layout_width="wrap_content"
                          android:layout_height="wrap_content"
                          android:id="@+id/container2a">
                <TextView
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:text="This is some text"
                        android:id="@+id/textview2"/>
            </LinearLayout>
        </LinearLayout>
        <LinearLayout android:orientation="vertical"
                      android:layout_width="wrap_content"
                      android:layout_height="wrap_content"
                      android:id="@+id/container3">
            <LinearLayout android:orientation="vertical"
                          android:layout_width="wrap_content"
                          android:layout_height="wrap_content"
                          android:id="@+id/container3a">
                <LinearLayout android:orientation="vertical"
                              android:layout_width="wrap_content"
                              android:layout_height="wrap_content"
                              android:id="@+id/container3b">
                    <TextView
                            android:layout_width="wrap_content"
                            android:layout_height="wrap_content"
                            android:text="This is some text"
                            android:id="@+id/textview3"/>
                </LinearLayout>
            </LinearLayout>
        </LinearLayout>
    </LinearLayout>
</LinearLayout>
 No newline at end of file
+162 −0
Original line number Diff line number Diff line
/*
 * 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 com.android.test.hwui;

import android.app.Activity;
import android.graphics.Color;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.LinearLayout;
import android.widget.TextView;

import java.util.ArrayList;

public class ViewLayerInvalidationActivity extends Activity {

    int currentColor = Color.WHITE;
    boolean nestedLayersOn = false;
    ArrayList<LinearLayout> linearLayouts = new ArrayList<LinearLayout>();
    ArrayList<LinearLayout> topLayouts = new ArrayList<LinearLayout>();
    ArrayList<TextView> textViews = new ArrayList<TextView>();
    LinearLayout container = null;
    boolean randomInvalidates = false;
    TextView nestedStatusTV, invalidateStatusTV;
    static final String NO_NESTING = "Nested Layer: NO   ";
    static final String NESTING = "Nested Layers: YES   ";
    static final String NO_INVALIDATING = "Random Invalidating: NO   ";
    static final String INVALIDATING = "Random Invalidating: YES   ";
    static final int TEXT_COLOR_INTERVAL = 400;
    static final int INVALIDATING_INTERVAL = 1000;
    static final int NESTING_INTERVAL = 2000;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.view_layer_invalidation);

        container = (LinearLayout) findViewById(R.id.container);
        final LinearLayout container1 = (LinearLayout) findViewById(R.id.container1);
        final LinearLayout container2 = (LinearLayout) findViewById(R.id.container2);
        final LinearLayout container3 = (LinearLayout) findViewById(R.id.container3);
        nestedStatusTV = (TextView) findViewById(R.id.nestedStatus);
        invalidateStatusTV = (TextView) findViewById(R.id.invalidateStatus);
        final TextView tva = (TextView) findViewById(R.id.textviewa);

        topLayouts.add(container1);
        topLayouts.add(container2);
        topLayouts.add(container3);

        collectLinearLayouts(container);
        collectTextViews(container);

        nestedStatusTV.setText(NO_NESTING);
        invalidateStatusTV.setText(NO_INVALIDATING);

        tva.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        container1.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        container2.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        container3.setLayerType(View.LAYER_TYPE_HARDWARE, null);

        container.postDelayed(textColorSetter, TEXT_COLOR_INTERVAL);
        container.postDelayed(nestedLayerSetter, NESTING_INTERVAL);
        container.postDelayed(randomInvalidatesSetter, INVALIDATING_INTERVAL);
    }

    private Runnable textColorSetter = new Runnable() {
        @Override
        public void run() {
            currentColor = (currentColor == Color.WHITE) ? Color.RED : Color.WHITE;
            for (TextView tv : textViews) {
                tv.setTextColor(currentColor);
            }
            if (randomInvalidates) {
                randomInvalidator(container);
            }
            container.postDelayed(textColorSetter, TEXT_COLOR_INTERVAL);
        }
    };

    private Runnable randomInvalidatesSetter = new Runnable() {
        @Override
        public void run() {
            randomInvalidates = !randomInvalidates;
            invalidateStatusTV.setText(randomInvalidates ? INVALIDATING : NO_INVALIDATING);
            container.postDelayed(randomInvalidatesSetter, INVALIDATING_INTERVAL);
        }
    };

    private Runnable nestedLayerSetter = new Runnable() {
        @Override
        public void run() {
            nestedLayersOn = !nestedLayersOn;
            nestedStatusTV.setText(nestedLayersOn ? NESTING : NO_NESTING);
            for (LinearLayout layout : linearLayouts) {
                layout.setLayerType(nestedLayersOn ?
                        View.LAYER_TYPE_HARDWARE : View.LAYER_TYPE_NONE, null);
            }
            if (!nestedLayersOn) {
                for (LinearLayout layout : topLayouts) {
                    layout.setLayerType(View.LAYER_TYPE_HARDWARE, null);
                }
            }
            container.postDelayed(nestedLayerSetter, NESTING_INTERVAL);
        }
    };

    /**
     * Invalidates views based on random chance (50%). This is meant to test
     * invalidating several items in the hierarchy at the same time, which can cause artifacts
     * if our invalidation-propagation logic is not sound.
     */
    private void randomInvalidator(ViewGroup parent) {
        for (int i = 0; i < parent.getChildCount(); ++i) {
            View child = parent.getChildAt(i);
            if (Math.random() < .5) {
                child.invalidate();
            }
            if (child instanceof ViewGroup) {
                randomInvalidator((ViewGroup) child);
            }
        }
    }

    private void collectLinearLayouts(View view) {
        if (!(view instanceof LinearLayout)) {
            return;
        }
        LinearLayout parent = (LinearLayout) view;
        linearLayouts.add(parent);
        for (int i = 0; i < parent.getChildCount(); ++i) {
            collectLinearLayouts(parent.getChildAt(i));
        }
    }

    private void collectTextViews(View view) {
        if (view instanceof TextView) {
            textViews.add((TextView) view);
            return;
        }
        if (!(view instanceof ViewGroup)) {
            return;
        }
        ViewGroup parent = (ViewGroup) view;
        for (int i = 0; i < parent.getChildCount(); ++i) {
            collectTextViews(parent.getChildAt(i));
        }
    }
}