代码之家  ›  专栏  ›  技术社区  ›  Sriram R

为同一图像的不同部分设置单击侦听器

  •  0
  • Sriram R  · 技术社区  · 6 年前

    我有一个像这样的单体图像

    enter image description here

    我需要做的是区分不同部位的身体点击。 例如,如果点击头部,则 onHeadClicked 应该执行。如果我点击左边,那么 onLeftHandClicked . 这会怎样?

    2 回复  |  直到 6 年前
        1
  •  1
  •   Marc Estrada    6 年前

    警告: 这个解决方案至少有5年的时间(我还在使用),所以肯定已经过时了,也许有新技术可以更好地解决这个问题。不管怎样,我会把解决方案贴在这里以帮助你。另外,这个解决方案有点麻烦,您可以重构或改进它,可以随意做。

    摘要

    此解决方案基于两个图像,其中一个图像可见(您要向用户显示哪个图像),另一个图像不可见。

    不可见图像的像素大小必须与目标图像的像素大小相同(在您的情况下为主体图像),将是透明的(或黑色背景),并且将用不同的颜色填充要单击的不同区域。

    所以,当你点击目标图像时,它的矩阵坐标会被选中,然后从这个坐标开始,按下的像素的颜色会从第二个图像中得到。

    由于您知道哪种颜色对应于主体的哪一部分(因为您以前已使用color->part_of_body配置了一个贴图),因此获取一种颜色可以准确地知道单击的主体部分。

    解决方案

    你有你的目标图像:

    enter image description here

    然后,需要创建第二个图像,如下所示:

    enter image description here

    现在你身体的每一部分都标上了不同的颜色。请注意,颜色明显不同。这就是为什么如果你使用相似的颜色,可能是冲突的时候,检索身体的一部分,因为颜色可以消除混淆。

    然后,您需要导出带有透明背景的彩色图像,您应该得到以下信息(请注意背景是透明的,但是stackoverflow的背景是白色的):

    enter image description here

    您将使用第一个和第三个图像,第二个图像只是中间步骤。

    首先,您必须在代码中配置地图颜色->正文部分:

    public HashMap<Integer, String> bodyParts;
    bodyParts.put(parseColor("#ff0000"), "part_1");
    bodyParts.put(parseColor("#00ff00"), "part_2");
    bodyParts.put(parseColor("#0000ff"), "part_3");
    bodyParts.put(parseColor("#ffff00"), "part_4");
    ... // Finish here with all your parts and colors
    

    然后我做了一个习惯 ImageView 为了更容易处理图像:

    ZoneTapimageView.java

        import android.annotation.SuppressLint;
        import android.content.Context;
        import android.graphics.Bitmap;
        import android.graphics.Color;
        import android.graphics.Matrix;
        import android.graphics.RectF;
        import android.graphics.drawable.BitmapDrawable;
        import android.graphics.drawable.Drawable;
        import android.util.AttributeSet;
        import android.util.Log;
        import android.view.Gravity;
        import android.view.View;
        import android.view.ViewGroup;
        import android.widget.FrameLayout;
        import android.widget.ImageView;
    
        import java.util.HashMap;
    
        import uk.co.senab.photoview.PhotoViewAttacher;
    
        public class ZoneTapImageView extends FrameLayout implements PhotoViewAttacher.OnPhotoTapListener {
          private static final String LOG_TAG = "ZoneMapTouch";
          private static final int DEFAULT_TOLERANCE = 25;
    
          private ImageView imageView;
          private ImageView imageViewAreas;
          @SuppressLint("UseSparseArrays")
          private HashMap<Integer, String> areasMap = new HashMap<>();
          private Context context;
          private OnAreaObtainedListener areaObtainedListener;
          private PhotoViewAttacher imageViewAttacher;
          private PhotoViewAttacher imageViewAreasAttacher;
    
          public ZoneTapImageView(Context context) {
            super(context);
            init(context);
          }
    
          public ZoneTapImageView(Context context, AttributeSet attrs) {
            super(context, attrs);
            init(context);
          }
    
          public ZoneTapImageView(Context context, AttributeSet attrs, int defStyle) {
            super(context, attrs, defStyle);
            init(context);
          }
    
          private void init(Context context) {
            int padding = getResources().getDimensionPixelSize(R.dimen.groups_padding_mini);
    
            this.context = context;
            FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
            this.setLayoutParams(params);
    
            params = new FrameLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT);
            params.gravity = Gravity.CENTER;
    
            imageView = new ImageView(getContext());
            imageView.setPadding(padding, padding, padding, padding);
            imageView.setLayoutParams(params);
            imageView.setScaleType(ImageView.ScaleType.CENTER_INSIDE);
            imageView.setAdjustViewBounds(true);
    
            imageViewAreas = new ImageView(getContext());
            imageViewAreas.setPadding(padding, padding, padding, padding);
            imageViewAreas.setLayoutParams(params);
            imageViewAreas.setScaleType(ImageView.ScaleType.CENTER_INSIDE);
            imageViewAreas.setAdjustViewBounds(true);
            imageViewAreas.setVisibility(View.INVISIBLE);
    
            this.addView(imageViewAreas);
            this.addView(imageView);
          }
    
          public void setImageResources(int resIdImage, int resIdImageAreas) {
            setImageResource(resIdImage);
            setImageResourceAreas(resIdImageAreas);
          }
    
          public void setImageResource(final int resIdImage) {
            imageView.post(new Runnable() {
              @Override
              public void run() {
                // Here I use a Image Cache, but it's not necessary
                ImageMemoryCache.loadBitmap(context, resIdImage, imageView, imageView.getWidth(), imageView.getHeight(), new ImageMemoryCache.OnImageLoadedListener() {
                  @Override
                  public void onImageLoaded() {
                    if (imageViewAttacher == null) {
                      imageViewAttacher = new PhotoViewAttacher(imageView);
                      imageViewAttacher.setZoomable(true);
                      imageViewAttacher.setOnPhotoTapListener(ZoneTapImageView.this);
                    }
                  }
                });
              }
            });
          }
    
          public void setImageResourceAreas(final int resIdImageAreas) {
            imageViewAreas.post(new Runnable() {
              @Override
              public void run() {
                // Here I use a Image Cache, but it's not necessary
                ImageMemoryCache.loadBitmap(context, resIdImageAreas, imageViewAreas, imageViewAreas.getWidth(), imageViewAreas.getHeight(), new ImageMemoryCache.OnImageLoadedListener() {
                  @Override
                  public void onImageLoaded() {
                    if (imageViewAreasAttacher == null) {
                      imageViewAreasAttacher = new PhotoViewAttacher(imageViewAreas);
                      imageViewAreasAttacher.setZoomable(false);
                    }
                  }
                });
              }
            });
          }
    
          public void setZoomOut() {
            if (imageViewAttacher != null)
              imageViewAttacher.setScale(1, true);
          }
    
          public void setOnAreaObtainedListener(OnAreaObtainedListener areaObtainedListener) {
            this.areaObtainedListener = areaObtainedListener;
          }
    
          @Override
          public void onPhotoTap(View view, float x, float y) {
            if (imageViewAreasAttacher == null) return;
            final RectF displayRect = imageViewAreasAttacher.getDisplayRect();
    
            float xAbsolute = x * displayRect.width() + displayRect.left;
            float yAbsolute = y * displayRect.height() + displayRect.top;
    
            Log.d("MapTouch", "X: " + xAbsolute + " Y: " + yAbsolute);
            getAreaFromCoordinatesAsync((int) xAbsolute, (int) yAbsolute, areaObtainedListener);
          }
    
          public void setAreasMap(HashMap<Integer, String> areasMap) {
            this.areasMap = areasMap;
          }
    
          public void getAreaFromCoordinatesAsync(final int x, final int y, final OnAreaObtainedListener onAreaObtainedListener) {
            new Thread(new Runnable() {
              @Override
              public void run() {
                String area = getAreaFromCoordinates(x, y);
                if (onAreaObtainedListener != null) {
                  onAreaObtainedListener.OnArea(area);
                }
              }
            }).start();
          }
    
          public String getAreaFromCoordinates(int x, int y) {
            int touchColor = getTapColor(x, y);
            Log.d(LOG_TAG, "Color (" + x + ", " + y + "): " + touchColor);
            if (touchColor == Integer.MIN_VALUE) return null;
            return getAreaFromColor(touchColor);
          }
    
          public String getAreaFromColor(int color) {
            for (Integer colorKey : areasMap.keySet()) {
              if (matchColor(colorKey, color, DEFAULT_TOLERANCE)) {
                return areasMap.get(colorKey);
              }
            }
            return null;
          }
    
          private boolean matchColor(int color1, int color2, int tolerance) {
            if (Math.abs(Color.red(color1) - Color.red(color2)) > tolerance)
              return false;
            if (Math.abs(Color.green(color1) - Color.green(color2)) > tolerance)
              return false;
            if (Math.abs(Color.blue(color1) - Color.blue(color2)) > tolerance)
              return false;
            return true;
          }
    
          private int getTapColor(int x, int y) {
            try {
              // Convert coordinates for scaled bitmap
              float[] eventXY = new float[]{x, y};
              Matrix invertMatrix = new Matrix();
              imageViewAreas.getImageMatrix().invert(invertMatrix);
              invertMatrix.mapPoints(eventXY);
              x = (int) eventXY[0];
              y = (int) eventXY[1];
    
              // Get bitmap
              Drawable imgDrawable = imageViewAreas.getDrawable();
              Bitmap bitmap = ((BitmapDrawable) imgDrawable).getBitmap();
    
              // Get color
              return bitmap.getPixel(x, y);
            } catch (Exception e) {
              return Integer.MIN_VALUE;
            }
          }
        }
    

    您需要以下依赖才能使其正常工作:

    compile 'com.github.chrisbanes.photoview:library:1.2.2'
    

    要实例化上一个类,可以执行以下操作:

    imageView = new ZoneTapImageView(getActivity());
    imageView.setImageResources(R.drawable.body_image, R.drawable.invisielb_areas_image);
    imageView.setAreasMap(bodyParts);
    imageView.setOnAreaObtainedListener(new OnAreaObtainedListener() {
      @Override
      public void OnArea(final String area) {
        Log.d("MapTouch", "Area:  " + area);
        // Area here is such "part_1" or "part_2"
        if (area != null) {
          // Handle your bodyPart click
        }
      }
    });
    

    仅此而已。希望我没有遗漏任何代码,希望这对你有帮助。

        2
  •  1
  •   Niza Siwale    6 年前

    首先,你必须在实际图像中存储身体点的坐标。然后简单地检查点击坐标是否在身体点的坐标范围内。您还应该确保使用imageview的矩阵和屏幕上的坐标来缩放这些点

    例如,如果

            class BodyPoint{
                  String name;
                  int x;
                  int y;
                  public(String name,int x,int y){
                       this.name = name;
                       this.x = x;
                       this.y = y;}
                 }
            BodyPoint headCoordinates = new BodyPoint ("head",50,20);
            BodyPoint neckCoordinates = new BodyPoint ("neck",50,50);
            BodyPoint leftHandCoordinates = new BodyPoint ("leftHand",10,50);
            BodyPoint rightHandCoordinates = new BodyPoint ("rightHand",80,50);
    
            BodyPoint[] bodyCoordinates = new BodyPoint[]{headCoordinates,neckCoordinates,
            leftHandCoordinates ,rightHandCoordinates };
    
        yourImgView.setOnTouchListener(new OnTouchListener() {
    
            @Override
            public boolean onTouch(View v, MotionEvent event) {
                 int[] viewCoords = new int[2];
                    yourImgView.getLocationOnScreen(viewCoords);
    
        int touchX = (int) event.getX();
        int touchY = (int) event.getY();
    
        int imageX = touchX - viewCoords[0]; // viewCoords[0] is the X coordinate
        int imageY = touchY - viewCoords[1]; 
        Matrix mMatrix = getImageMatrix();
        float[] f = new float[9];
        mMatrix .getValues(f);
    
       float scaleX = f[Matrix.MSCALE_X];
       float scaleY = f[Matrix.MSCALE_Y];
    
        processTouchedPoints(imageX/(int)scaleX , imageY/(int)scaleY );
                return true;
    
            }
        });
    
        ...
        int range = 50;
        void processTouchedPoints(int imageX,int imageY){
          foreach(BodyPoint mBodyPoint:bodyCoordinates ){
        int x = mBodyPoint.x;
        int y = mBodyPoint.y;
              if((imageX> (x-range) && imageX<(x+range))
               &&(imageY> (y-range) && imageY<(y+range))){
               doWhatever(mBodyPoint.name)
             }
          }
        }