2017-05-15 11 views
0

내 앱에서 한 개의 이미지보기 만 긴 세로 이미지를 보여 주므로 이미지를 확대/축소 할 수 있습니다 이 enter image description hereAndroid Studio에서 확대/축소 및 화면 이동과 함께 화면의 너비를 화면에 맞출 수있는 방법

하지만 내가 정말 원하는 것은 내 이미지가 화면의 너비에 맞게 이미지 사용자 아래로 더 이동하는 것입니다 :하지만 내 문제는 내가 이미지의 전체 길이는 다음과 같이 표시되는 활동을 열 때입니다 다음과 같이 스크롤하십시오 enter image description here

이 자바 클래스에는 확대/축소 및 패닝을위한 코드가 포함되어 있습니다.

import android.content.Context; 
import android.graphics.Matrix; 
import android.graphics.PointF; 
import android.graphics.drawable.Drawable; 
import android.util.AttributeSet; 
import android.util.Log; 
import android.view.MotionEvent; 
import android.view.ScaleGestureDetector; 
import android.view.View; 
import android.widget.ImageView; 

public class TouchImageView extends ImageView { 
    Matrix matrix; 
    // We can be in one of these 3 states 
    static final int NONE = 0; 
    static final int DRAG = 1; 
    static final int ZOOM = 2; 
    int mode = NONE; 
    // Remember some things for zooming 
    PointF last = new PointF(); 
    PointF start = new PointF(); 
    float minScale = 1f; 
    float maxScale = 3f; 
    float[] m; 
    int viewWidth, viewHeight; 
    static final int CLICK = 3; 
    float saveScale = 1f; 
    protected float origWidth, origHeight; 
    int oldMeasuredWidth, oldMeasuredHeight; 
    ScaleGestureDetector mScaleDetector; 
    Context context; 

    public TouchImageView(Context context) { 
     super(context); 
     sharedConstructing(context); 
    } 

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

    private void sharedConstructing(Context context) { 
     super.setClickable(true); 
     this.context = context; 
     mScaleDetector = new ScaleGestureDetector(context, new ScaleListener()); 
     matrix = new Matrix(); 
     m = new float[9]; 
     setImageMatrix(matrix); 
     setScaleType(ScaleType.MATRIX); 
     setOnTouchListener(new OnTouchListener() { 

      @Override 
      public boolean onTouch(View v, MotionEvent event) { 
       mScaleDetector.onTouchEvent(event); 
       PointF curr = new PointF(event.getX(), event.getY()); 
       switch (event.getAction()) { 
        case MotionEvent.ACTION_DOWN: 
         last.set(curr); 
         start.set(last); 
         mode = DRAG; 
         break; 
        case MotionEvent.ACTION_MOVE: 
         if (mode == DRAG) { 
          float deltaX = curr.x - last.x; 
          float deltaY = curr.y - last.y; 
          float fixTransX = getFixDragTrans(deltaX, viewWidth, origWidth * saveScale); 
          float fixTransY = getFixDragTrans(deltaY, viewHeight, origHeight * saveScale); 
          matrix.postTranslate(fixTransX, fixTransY); 
          fixTrans(); 
          last.set(curr.x, curr.y); 
         } 
         break; 
        case MotionEvent.ACTION_UP: 
         mode = NONE; 
         int xDiff = (int) Math.abs(curr.x - start.x); 
         int yDiff = (int) Math.abs(curr.y - start.y); 
         if (xDiff < CLICK && yDiff < CLICK) 
          performClick(); 
         break; 
        case MotionEvent.ACTION_POINTER_UP: 
         mode = NONE; 
         break; 
       } 
       setImageMatrix(matrix); 
       invalidate(); 
       return true; // indicate event was handled 
      } 
     }); 
    } 

    public void setMaxZoom(float x) { 
     maxScale = x; 
    } 

    private class ScaleListener extends ScaleGestureDetector.SimpleOnScaleGestureListener { 
     @Override 
     public boolean onScaleBegin(ScaleGestureDetector detector) { 
      mode = ZOOM; 
      return true; 
     } 

     @Override 
     public boolean onScale(ScaleGestureDetector detector) { 
      float mScaleFactor = detector.getScaleFactor(); 
      float origScale = saveScale; 
      saveScale *= mScaleFactor; 
      if (saveScale > maxScale) { 
       saveScale = maxScale; 
       mScaleFactor = maxScale/origScale; 
      } else if (saveScale < minScale) { 
       saveScale = minScale; 
       mScaleFactor = minScale/origScale; 
      } 
      if (origWidth * saveScale <= viewWidth || origHeight * saveScale <= viewHeight) 
       matrix.postScale(mScaleFactor, mScaleFactor, viewWidth/2, viewHeight/2); 
      else 
       matrix.postScale(mScaleFactor, mScaleFactor, detector.getFocusX(), detector.getFocusY()); 
      fixTrans(); 
      return true; 
     } 
    } 

    void fixTrans() { 
     matrix.getValues(m); 
     float transX = m[Matrix.MTRANS_X]; 
     float transY = m[Matrix.MTRANS_Y]; 
     float fixTransX = getFixTrans(transX, viewWidth, origWidth * saveScale); 
     float fixTransY = getFixTrans(transY, viewHeight, origHeight * saveScale); 
     if (fixTransX != 0 || fixTransY != 0) 
      matrix.postTranslate(fixTransX, fixTransY); 
    } 

    float getFixTrans(float trans, float viewSize, float contentSize) { 
     float minTrans, maxTrans; 
     if (contentSize <= viewSize) { 
      minTrans = 0; 
      maxTrans = viewSize - contentSize; 
     } else { 
      minTrans = viewSize - contentSize; 
      maxTrans = 0; 
     } 
     if (trans < minTrans) 
      return -trans + minTrans; 
     if (trans > maxTrans) 
      return -trans + maxTrans; 
     return 0; 
    } 

    float getFixDragTrans(float delta, float viewSize, float contentSize) { 
     if (contentSize <= viewSize) { 
      return 0; 
     } 
     return delta; 
    } 

    @Override 
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
     super.onMeasure(widthMeasureSpec, heightMeasureSpec); 
     viewWidth = MeasureSpec.getSize(widthMeasureSpec); 
     viewHeight = MeasureSpec.getSize(heightMeasureSpec); 
     // 
     // Rescales image on rotation 
     // 
     if (oldMeasuredHeight == viewWidth && oldMeasuredHeight == viewHeight || viewWidth == 0 || viewHeight == 0) 
      return; 
     oldMeasuredHeight = viewHeight; 
     oldMeasuredWidth = viewWidth; 
     if (saveScale == 1) { 
      //Fit to screen. 
      float scale; 
      Drawable drawable = getDrawable(); 
      if (drawable == null || drawable.getIntrinsicWidth() == 0 || drawable.getIntrinsicHeight() == 0) 
       return; 
      int bmWidth = drawable.getIntrinsicWidth(); 
      int bmHeight = drawable.getIntrinsicHeight(); 
      Log.d("bmSize", "bmWidth: " + bmWidth + " bmHeight : " + bmHeight); 
      float scaleX = (float) viewWidth/(float) bmWidth; 
      float scaleY = (float) viewHeight/(float) bmHeight; 
      scale = Math.min(scaleX, scaleY); 
      matrix.setScale(scale, scale); 
      // Center the image 
      float redundantYSpace = (float) viewHeight - (scale * (float) bmHeight); 
      float redundantXSpace = (float) viewWidth - (scale * (float) bmWidth); 
      redundantYSpace /= (float) 2; 
      redundantXSpace /= (float) 2; 
      matrix.postTranslate(redundantXSpace, redundantYSpace); 
      origWidth = viewWidth - 2 * redundantXSpace; 
      origHeight = viewHeight - 2 * redundantYSpace; 
      setImageMatrix(matrix); 
     } 
     fixTrans(); 
    } 

}` 

이 클래스는 화면의 이미지를 보여줍니다

import android.os.Bundle; 
import android.app.Activity; 
import android.view.Menu; 

import static android.R.attr.button; 

public class MultiTouchActivity extends Activity { 
    @Override 
    protected void onCreate(Bundle savedInstanceState) { 
     super.onCreate(savedInstanceState); 
     TouchImageView img = new TouchImageView(this); 
     img.setImageResource(R.drawable.pg2); 
     img.setMaxZoom(4f); 
     setContentView(img); 
    } 
} 

나는 길이보다는 화면에 이미지의 너비에 맞게 내 코드에서 확인해야합니다 및 변경되는지 알려주세요 여전히 이미지를 확대/축소하고 팬 할 수 있어야합니다.

+0

setScaleType (ScaleType.MATRIX) :이 방법을

TouchImageView img = (TouchImageView)findViewById(R.id.img); 

을 그리고 오버라이드 (override) :

<ScrollView xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="wrap_content" android:scrollbarAlwaysDrawVerticalTrack="true" > <TouchImageView android:id="@+id/img" android:contentDescription="Specs" android:layout_width="match_parent" android:layout_height="wrap_content" android:src="@drawable/specs" > </TouchImageView > 

활동이 레이아웃을 부풀려과 같은 그에서 TouchImageView를 얻을 수 fitxy와 xml보기 너비 일치 부모와 높이 wrapcontent – jagapathi

답변

0

ScrollView를 사용하여 예상되는 동작을 얻을 수 있습니다.

@Override 
public boolean dispatchTouchEvent(MotionEvent ev) { 
    super.dispatchTouchEvent(ev); 
    return image.dispatchTouchEvent(ev); 
} 
+0

코드를 변경 한 후 화면의 왼쪽에서만 확대되어 이미지 전체에 팬을 수행 할 수 없습니다. 어디서나 이미지를 확대/축소하는 방법을 알려주십시오. 그리고 그것도 가로 질러 팬 감사합니다 – Rumaisa

+0

런타임 오류가 발생했습니다 이 줄에 오류가 있습니다 : "TouchImageView img = (TouchImageView) findViewById (R.id.img);" 안드로이드 모니터 오류 메시지가 "com.example.android.multitouchpinchzoompan.MultiTouchActivity에서. java.lang.reflect.Constructor.newInstance에서 (MultiTouchActivity.java:12) (기본 방법)"을 읽고 무슨 잘못 내가 그나마 는 알고있다. 지원해 주셔서 감사합니다. – Rumaisa

+0

문제를 식별 할 수 있도록 활동 코드를 게시하십시오. – Sac