/* * Copyright (C) 2016 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.support.transition; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.ObjectAnimator; import android.animation.PropertyValuesHolder; import android.annotation.TargetApi; import android.graphics.Bitmap; import android.graphics.Canvas; import android.graphics.Rect; import android.graphics.drawable.BitmapDrawable; import android.support.annotation.RequiresApi; import android.view.View; import android.view.ViewGroup; import java.util.Map; /** * This transition captures the layout bounds of target views before and after * the scene change and animates those changes during the transition. * * <p>A ChangeBounds transition can be described in a resource file by using the * tag <code>changeBounds</code>, along with the other standard * attributes of {@link android.R.styleable#Transition}.</p> */ @RequiresApi(14) @TargetApi(14) class ChangeBoundsPort extends TransitionPort { private static final String PROPNAME_BOUNDS = "android:changeBounds:bounds"; private static final String PROPNAME_PARENT = "android:changeBounds:parent"; private static final String PROPNAME_WINDOW_X = "android:changeBounds:windowX"; private static final String PROPNAME_WINDOW_Y = "android:changeBounds:windowY"; private static final String[] sTransitionProperties = { PROPNAME_BOUNDS, PROPNAME_PARENT, PROPNAME_WINDOW_X, PROPNAME_WINDOW_Y }; private static final String LOG_TAG = "ChangeBounds"; private static RectEvaluator sRectEvaluator = new RectEvaluator(); int[] tempLocation = new int[2]; boolean mResizeClip = false; boolean mReparent = false; @Override public String[] getTransitionProperties() { return sTransitionProperties; } public void setResizeClip(boolean resizeClip) { mResizeClip = resizeClip; } /** * Setting this flag tells ChangeBounds to track the before/after parent * of every view using this transition. The flag is not enabled by * default because it requires the parent instances to be the same * in the two scenes or else all parents must use ids to allow * the transition to determine which parents are the same. * * @param reparent true if the transition should track the parent * container of target views and animate parent changes. */ public void setReparent(boolean reparent) { mReparent = reparent; } private void captureValues(TransitionValues values) { View view = values.view; values.values.put(PROPNAME_BOUNDS, new Rect(view.getLeft(), view.getTop(), view.getRight(), view.getBottom())); values.values.put(PROPNAME_PARENT, values.view.getParent()); values.view.getLocationInWindow(tempLocation); values.values.put(PROPNAME_WINDOW_X, tempLocation[0]); values.values.put(PROPNAME_WINDOW_Y, tempLocation[1]); } @Override public void captureStartValues(TransitionValues transitionValues) { captureValues(transitionValues); } @Override public void captureEndValues(TransitionValues transitionValues) { captureValues(transitionValues); } @Override public Animator createAnimator(final ViewGroup sceneRoot, TransitionValues startValues, TransitionValues endValues) { if (startValues == null || endValues == null) { return null; } Map<String, Object> startParentVals = startValues.values; Map<String, Object> endParentVals = endValues.values; ViewGroup startParent = (ViewGroup) startParentVals.get(PROPNAME_PARENT); ViewGroup endParent = (ViewGroup) endParentVals.get(PROPNAME_PARENT); if (startParent == null || endParent == null) { return null; } final View view = endValues.view; boolean parentsEqual = (startParent == endParent) || (startParent.getId() == endParent.getId()); // TODO: Might want reparenting to be separate/subclass transition, or at least // triggered by a property on ChangeBounds. Otherwise, we're forcing the requirement that // all parents in layouts have IDs to avoid layout-inflation resulting in a side-effect // of reparenting the views. if (!mReparent || parentsEqual) { Rect startBounds = (Rect) startValues.values.get(PROPNAME_BOUNDS); Rect endBounds = (Rect) endValues.values.get(PROPNAME_BOUNDS); int startLeft = startBounds.left; int endLeft = endBounds.left; int startTop = startBounds.top; int endTop = endBounds.top; int startRight = startBounds.right; int endRight = endBounds.right; int startBottom = startBounds.bottom; int endBottom = endBounds.bottom; int startWidth = startRight - startLeft; int startHeight = startBottom - startTop; int endWidth = endRight - endLeft; int endHeight = endBottom - endTop; int numChanges = 0; if (startWidth != 0 && startHeight != 0 && endWidth != 0 && endHeight != 0) { if (startLeft != endLeft) { ++numChanges; } if (startTop != endTop) { ++numChanges; } if (startRight != endRight) { ++numChanges; } if (startBottom != endBottom) { ++numChanges; } } if (numChanges > 0) { if (!mResizeClip) { PropertyValuesHolder pvh[] = new PropertyValuesHolder[numChanges]; int pvhIndex = 0; if (startLeft != endLeft) { view.setLeft(startLeft); } if (startTop != endTop) { view.setTop(startTop); } if (startRight != endRight) { view.setRight(startRight); } if (startBottom != endBottom) { view.setBottom(startBottom); } if (startLeft != endLeft) { pvh[pvhIndex++] = PropertyValuesHolder.ofInt("left", startLeft, endLeft); } if (startTop != endTop) { pvh[pvhIndex++] = PropertyValuesHolder.ofInt("top", startTop, endTop); } if (startRight != endRight) { pvh[pvhIndex++] = PropertyValuesHolder.ofInt("right", startRight, endRight); } if (startBottom != endBottom) { pvh[pvhIndex++] = PropertyValuesHolder.ofInt("bottom", startBottom, endBottom); } ObjectAnimator anim = ObjectAnimator.ofPropertyValuesHolder(view, pvh); if (view.getParent() instanceof ViewGroup) { final ViewGroup parent = (ViewGroup) view.getParent(); // parent.suppressLayout(true); TransitionListener transitionListener = new TransitionListenerAdapter() { boolean mCanceled = false; @Override public void onTransitionCancel(TransitionPort transition) { // parent.suppressLayout(false); mCanceled = true; } @Override public void onTransitionEnd(TransitionPort transition) { if (!mCanceled) { // parent.suppressLayout(false); } } @Override public void onTransitionPause(TransitionPort transition) { // parent.suppressLayout(false); } @Override public void onTransitionResume(TransitionPort transition) { // parent.suppressLayout(true); } }; addListener(transitionListener); } return anim; } else { if (startWidth != endWidth) { view.setRight(endLeft + Math.max(startWidth, endWidth)); } if (startHeight != endHeight) { view.setBottom(endTop + Math.max(startHeight, endHeight)); } // TODO: don't clobber TX/TY if (startLeft != endLeft) { view.setTranslationX(startLeft - endLeft); } if (startTop != endTop) { view.setTranslationY(startTop - endTop); } // Animate location with translationX/Y and size with clip bounds float transXDelta = endLeft - startLeft; float transYDelta = endTop - startTop; int widthDelta = endWidth - startWidth; int heightDelta = endHeight - startHeight; numChanges = 0; if (transXDelta != 0) { numChanges++; } if (transYDelta != 0) { numChanges++; } if (widthDelta != 0 || heightDelta != 0) { numChanges++; } PropertyValuesHolder pvh[] = new PropertyValuesHolder[numChanges]; int pvhIndex = 0; if (transXDelta != 0) { pvh[pvhIndex++] = PropertyValuesHolder.ofFloat("translationX", view.getTranslationX(), 0); } if (transYDelta != 0) { pvh[pvhIndex++] = PropertyValuesHolder.ofFloat("translationY", view.getTranslationY(), 0); } if (widthDelta != 0 || heightDelta != 0) { Rect tempStartBounds = new Rect(0, 0, startWidth, startHeight); Rect tempEndBounds = new Rect(0, 0, endWidth, endHeight); // pvh[pvhIndex++] = PropertyValuesHolder.ofObject("clipBounds", // sRectEvaluator, tempStartBounds, tempEndBounds); } ObjectAnimator anim = ObjectAnimator.ofPropertyValuesHolder(view, pvh); if (view.getParent() instanceof ViewGroup) { final ViewGroup parent = (ViewGroup) view.getParent(); // parent.suppressLayout(true); TransitionListener transitionListener = new TransitionListenerAdapter() { boolean mCanceled = false; @Override public void onTransitionCancel(TransitionPort transition) { // parent.suppressLayout(false); mCanceled = true; } @Override public void onTransitionEnd(TransitionPort transition) { if (!mCanceled) { // parent.suppressLayout(false); } } @Override public void onTransitionPause(TransitionPort transition) { // parent.suppressLayout(false); } @Override public void onTransitionResume(TransitionPort transition) { // parent.suppressLayout(true); } }; addListener(transitionListener); } anim.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { // view.setClipBounds(null); } }); return anim; } } } else { int startX = (Integer) startValues.values.get(PROPNAME_WINDOW_X); int startY = (Integer) startValues.values.get(PROPNAME_WINDOW_Y); int endX = (Integer) endValues.values.get(PROPNAME_WINDOW_X); int endY = (Integer) endValues.values.get(PROPNAME_WINDOW_Y); // TODO: also handle size changes: check bounds and animate size changes if (startX != endX || startY != endY) { sceneRoot.getLocationInWindow(tempLocation); Bitmap bitmap = Bitmap.createBitmap(view.getWidth(), view.getHeight(), Bitmap.Config.ARGB_8888); Canvas canvas = new Canvas(bitmap); view.draw(canvas); final BitmapDrawable drawable = new BitmapDrawable(bitmap); view.setVisibility(View.INVISIBLE); ViewOverlay.createFrom(sceneRoot).add(drawable); // sceneRoot.getOverlay().add(drawable); Rect startBounds1 = new Rect(startX - tempLocation[0], startY - tempLocation[1], startX - tempLocation[0] + view.getWidth(), startY - tempLocation[1] + view.getHeight()); Rect endBounds1 = new Rect(endX - tempLocation[0], endY - tempLocation[1], endX - tempLocation[0] + view.getWidth(), endY - tempLocation[1] + view.getHeight()); ObjectAnimator anim = ObjectAnimator.ofObject(drawable, "bounds", sRectEvaluator, startBounds1, endBounds1); anim.addListener(new AnimatorListenerAdapter() { @Override public void onAnimationEnd(Animator animation) { ViewOverlay.createFrom(sceneRoot).remove(drawable); // sceneRoot.getOverlay().remove(drawable); view.setVisibility(View.VISIBLE); } }); return anim; } } return null; } }