警告:
这个解决方案至少有5年的时间(我还在使用),所以肯定已经过时了,也许有新技术可以更好地解决这个问题。不管怎样,我会把解决方案贴在这里以帮助你。另外,这个解决方案有点麻烦,您可以重构或改进它,可以随意做。
摘要
此解决方案基于两个图像,其中一个图像可见(您要向用户显示哪个图像),另一个图像不可见。
不可见图像的像素大小必须与目标图像的像素大小相同(在您的情况下为主体图像),将是透明的(或黑色背景),并且将用不同的颜色填充要单击的不同区域。
所以,当你点击目标图像时,它的矩阵坐标会被选中,然后从这个坐标开始,按下的像素的颜色会从第二个图像中得到。
由于您知道哪种颜色对应于主体的哪一部分(因为您以前已使用color->part_of_body配置了一个贴图),因此获取一种颜色可以准确地知道单击的主体部分。
解决方案
你有你的目标图像:
然后,需要创建第二个图像,如下所示:
现在你身体的每一部分都标上了不同的颜色。请注意,颜色明显不同。这就是为什么如果你使用相似的颜色,可能是冲突的时候,检索身体的一部分,因为颜色可以消除混淆。
然后,您需要导出带有透明背景的彩色图像,您应该得到以下信息(请注意背景是透明的,但是stackoverflow的背景是白色的):
您将使用第一个和第三个图像,第二个图像只是中间步骤。
首先,您必须在代码中配置地图颜色->正文部分:
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
}
}
});
仅此而已。希望我没有遗漏任何代码,希望这对你有帮助。