捏放大MyTextureView使其仅缩放到屏幕尺寸(宽度,高度)

时间:2018-11-04 21:13:05

标签: android surfaceview pinchzoom android-framelayout android-textureview

最近,我使用一种实现方式来使用SurfaceView缩小我的TextureView。缩放效果完美,但是视图不会扩大到屏幕尺寸。在我的情况下,TextureView的尺寸始终为1080、612。我想修改此缩放比例,以便在进行捏合时视图可以填满整个屏幕。

这是我的接收TextureView的类的代码:

public class ScaleManager extends BaseInputManager {

    private final TextureView mTargetView;
    private final ScaleGestureDetector mScaleDetector;
    private static final float MIN_SCALE = 1.0f;
    private static final float MAX_SCALE = 5.0f;
    private float mCurrentScale = MIN_SCALE;
    private Matrix mMatrix = new Matrix();
    private float[] mMatrixValues = new float[9];
    private float mPreviousX;
    private float mPreviousY;
    private boolean mIsDragInProgress = false;

    public ScaleManager(TextureView view) {
        mTargetView = view;
        mScaleDetector = new ScaleGestureDetector(view.getContext(), new ScaleListener());
        mScaleDetector.setQuickScaleEnabled(false);
    }


    @Override
    public boolean handle(KeyEvent event) {
        return false;
    }

    @Override
    public boolean handle(MotionEvent event) {
        final int index = event.getActionIndex();
        final int type = event.getToolType(index);
        if (type == MotionEvent.TOOL_TYPE_FINGER || type == MotionEvent.TOOL_TYPE_STYLUS) {
            mScaleDetector.onTouchEvent(event);

            mMatrix.getValues(mMatrixValues);
            boolean isScaling = (mMatrixValues[Matrix.MSCALE_X] != 1.0f && mMatrixValues[Matrix.MSCALE_Y] != 1.0f);
            if (isScaling || mScaleDetector.isInProgress()) {
                switch (event.getAction()) {
                    case MotionEvent.ACTION_POINTER_DOWN:
                    case MotionEvent.ACTION_DOWN:
                        if (event.getPointerCount() <= 2) {
                            mPreviousX = event.getX();
                            mPreviousY = event.getY();
                        }

                        if (event.getPointerCount() == 2) {
                            mIsDragInProgress = true;
                        }
                        break;
                    case MotionEvent.ACTION_MOVE:
                        if (event.getPointerCount() == 2) {
                            float x = mMatrixValues[Matrix.MTRANS_X];
                            float y = mMatrixValues[Matrix.MTRANS_Y];

                            float width = mTargetView.getWidth();
                            float height = mTargetView.getHeight();

                            float right = width * mCurrentScale - width;
                            float bottom = height * mCurrentScale - height;

                            float deltaX = event.getX() - mPreviousX;// x difference
                            float deltaY = event.getY() - mPreviousY;// y difference

                            float scaledWidth = Math.round(width * mCurrentScale);// width after applying current scale
                            float scaledHeight = Math.round(height * mCurrentScale);// height after applying current scale
                            //if scaledWidth is smaller than the views width
                            //in other words if the image width fits in the view
                            //limit left and right movement
                            if (scaledWidth < width) {
                                deltaX = 0;
                                if (y + deltaY > 0) {
                                    deltaY = -y;
                                } else if (y + deltaY < -bottom) {
                                    deltaY = -(y + bottom);
                                }

                                //if scaledHeight is smaller than the views height
                                //in other words if the image height fits in the view
                                //limit up and down movement
                            } else if (scaledHeight < height) {
                                deltaY = 0;
                                if (x + deltaX > 0) {
                                    deltaX = -x;
                                } else if (x + deltaX < -right) {
                                    deltaX = -(x + right);
                                }

                                //if the image doesn't fit in the width or height
                                //limit both up and down and left and right
                            } else {
                                if (x + deltaX > 0) {
                                    deltaX = -x;
                                } else if (x + deltaX < -right) {
                                    deltaX = -(x + right);
                                }

                                if (y + deltaY > 0) {
                                    deltaY = -y;
                                } else if (y + deltaY < -bottom) {
                                    deltaY = -(y + bottom);
                                }
                            }

                            //move the image with the matrix
                            mMatrix.postTranslate(deltaX, deltaY);


                            mTargetView.setTransform(mMatrix);
                            mTargetView.invalidate();

                            mPreviousX = event.getX();
                            mPreviousY = event.getY();
                        }
                        break;
                    case MotionEvent.ACTION_POINTER_UP:

                    case MotionEvent.ACTION_UP:
                        if (event.getPointerCount() == 1) {
                            mIsDragInProgress = false;
                        }
                        break;
                }
            }


            return mScaleDetector.isInProgress() || mIsDragInProgress;
        }

        return false;
    }


    private class ScaleListener implements ScaleGestureDetector.OnScaleGestureListener {

        @Override
        public boolean onScaleBegin(ScaleGestureDetector detector) {
            return true;
        }

        @Override
        public boolean onScale(ScaleGestureDetector detector) {
            float width = mTargetView.getWidth();
            float height = mTargetView.getHeight();
            float factor = detector.getScaleFactor();
            float scale = mCurrentScale * factor;
            if (scale > MAX_SCALE) {
                factor = MAX_SCALE / mCurrentScale;
                mCurrentScale = MAX_SCALE;
            } else if (scale < MIN_SCALE) {
                factor = MIN_SCALE / mCurrentScale;
                mCurrentScale = MIN_SCALE;
            } else {
                mCurrentScale = scale;
            }

            if (width * scale <= width || height * scale <= height) {
                mMatrix.postScale(factor, factor, width / 2, height / 2);
            } else {
                mMatrix.postScale(factor, factor, detector.getFocusX(), detector.getFocusY());
            }

            if (factor < 1) {
                float right = width * mCurrentScale - width;
                float bottom = height * mCurrentScale - height;

                mMatrix.getValues(mMatrixValues);
                float x = mMatrixValues[Matrix.MTRANS_X];
                float y = mMatrixValues[Matrix.MTRANS_Y];

                if (x < -right) {
                    mMatrix.postTranslate(-(x + right), 0);
                } else if (x > 0) {
                    mMatrix.postTranslate(-x, 0);
                }

                if (y < -bottom) {
                    mMatrix.postTranslate(0, -(y + bottom));
                } else if (y > 0) {
                    mMatrix.postTranslate(0, -y);
                }
            }



            return true;
        }

        @Override
        public void onScaleEnd(ScaleGestureDetector detector) {
        }
    }


}

启动视图时,我使用名为AspectFrameLayout的类来确定视图的初始比例。每次修改视图以保持纵横比时,都会调用此AspectFrameLayout。请遵循代码:

<com.examples.views.AspectRatioFrameLayout
    android:id="@+id/player_video"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_gravity="center"
    android:focusable="true"
    android:focusableInTouchMode="true" />




mSurfaceView = layoutInflater.inflate(R.layout.texture_view, 
player_video_frame, false)

这是我用来保持长宽比的AspectFrameLayout类:

public final class AspectRatioFrameLayout extends FrameLayout {


    private static final float MAX_ASPECT_RATIO_DEFORMATION_FRACTION = 0.01f;

    private float mVideoAspectRatio;

    public AspectRatioFrameLayout(Context context) {
        super(context);
    }

    public AspectRatioFrameLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
    }


    public void setAspectRatio(float ratio) {
        if (this.mVideoAspectRatio != ratio) {
            this.mVideoAspectRatio = ratio;
            requestLayout();
        }
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        if (mVideoAspectRatio == 0) {
            // Aspect ratio not set.
            return;
        }

        int width = getMeasuredWidth();
        int height = getMeasuredHeight();
        float viewAspectRatio = (float) width / height;
        float aspectDeformation = mVideoAspectRatio / viewAspectRatio - 1;
        if (Math.abs(aspectDeformation) <= MAX_ASPECT_RATIO_DEFORMATION_FRACTION) {
            // We're within the allowed tolerance.
            return;
        }

        if (aspectDeformation > 0) {
            height = (int) (width / mVideoAspectRatio);
        } else {
            width = (int) (height * mVideoAspectRatio);
        }
        super.onMeasure(
                MeasureSpec.makeMeasureSpec(width, MeasureSpec.EXACTLY),
                MeasureSpec.makeMeasureSpec(height, MeasureSpec.EXACTLY)
        );
    }

}

0 个答案:

没有答案