Android实现拖拽GridView到目标View

简介:
package cn.com.weather.serviceDemo.view;

import cn.com.weather.serviceDemo.interfaces.OnDropListener;
import cn.com.weather.serviceDemo.util.DeviceUtil;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Matrix;
import android.graphics.PixelFormat;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.AdapterView;
import android.widget.GridView;
import android.widget.ImageView;

public class MyGridView extends GridView{
	
	private ImageView dragImageView;
	private WindowManager windowManager;
	private WindowManager.LayoutParams windowParams;
	
	private int dragSrcPosition;//开始拖拽项
	private int dragPosition;
	
	//移动的位置
    private int dragPointX;
    private int dragPointY;
    //当前位置距离边界的位置
    private int dragOffsetX;
    private int dragOffsetY;
    
    private View myView;//目标view
    private int dropPosition = -1;//选中项
    
    private OnDropListener onDropListener;

	public MyGridView(Context context, AttributeSet attrs) {
		super(context, attrs);
	}
	
	public void setView(View view){
		myView = view;
	}
	
	public void setOnDropListener(OnDropListener onDropListener){
		this.onDropListener = onDropListener;
	}
	
	private boolean setOnItemLongClickListener(final MotionEvent ev){
		this.setOnItemLongClickListener(new OnItemLongClickListener() {

			@Override
			public boolean onItemLongClick(AdapterView<?> parent, View view,
					int position, long id) {
				final int x = (int)ev.getX();//触摸点相对于GridView的坐标 
				final int y = (int)ev.getY();
				
				dragSrcPosition = dragPosition = position;
				
				if(dragPosition == AdapterView.INVALID_POSITION ){
					return false;
				}
				
				final ViewGroup item = (ViewGroup)getChildAt(dragPosition - getFirstVisiblePosition());
				
				dragPointX = x - item.getLeft();
				dragPointY = y - item.getTop();
				dragOffsetX = (int)ev.getRawX() - x;
				dragOffsetY = (int)ev.getRawY() - y;
				
				item.destroyDrawingCache();//清除cache
				item.setDrawingCacheEnabled(true);
				Bitmap bitmap = Bitmap.createBitmap(item.getDrawingCache());
				startDrag(getScaleBitmap(bitmap), x, y);
				DeviceUtil.Vibrate(getContext(), 100);
				return false;
			}
		});
		
		return super.onInterceptTouchEvent(ev);
	}

	@Override
	public boolean onInterceptTouchEvent(MotionEvent ev) {
		if(ev.getAction() == MotionEvent.ACTION_DOWN){
			return setOnItemLongClickListener(ev);
		}
		return super.onInterceptTouchEvent(ev);
	}
	
	@Override
	public boolean onTouchEvent(MotionEvent ev) {
		if(dragImageView != null && dragPosition != AdapterView.INVALID_POSITION){
			int x = (int)ev.getX();
			int y = (int)ev.getY();
			switch (ev.getAction()) {
			case MotionEvent.ACTION_UP:
				stopDrag();
				onDrop((int)ev.getRawX(), (int)ev.getRawY());
				break;

			case MotionEvent.ACTION_MOVE:
				onDrag(x, y);
				break;
			}
			
			
		}
		return super.onTouchEvent(ev);
	}

	private void startDrag(Bitmap bm, int x, int y){
		stopDrag();
		
		windowParams = new WindowManager.LayoutParams();
		windowParams.gravity = Gravity.TOP|Gravity.LEFT;
		//左上角相对于屏幕的坐标
		windowParams.x = x - dragPointX + dragOffsetX;
		windowParams.y = y - dragPointY + dragOffsetY;
		
		windowParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
		windowParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
		windowParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
								| WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE
								| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON
								| WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN;
		windowParams.format = PixelFormat.TRANSLUCENT;
		windowParams.windowAnimations = 0;
		
		ImageView imageView = new ImageView(this.getContext());
		imageView.setImageBitmap(bm);
		
		windowManager = (WindowManager)this.getContext().getSystemService(Context.WINDOW_SERVICE);
		windowManager.addView(imageView, windowParams);
		dragImageView = imageView;
	}
	
	private void stopDrag(){
		if (dragImageView != null) {
			windowManager.removeView(dragImageView);
			dragImageView = null;
		}
	}
	
	private void onDrag(int x, int y){
		if(dragImageView != null){
			windowParams.alpha = 0.8f;
			windowParams.x = x - dragPointX + dragOffsetX;
			windowParams.y = y - dragPointY + dragOffsetY;
			windowManager.updateViewLayout(dragImageView, windowParams);
		}
		
	}
	
	private void onDrop(int x, int y){
		boolean inView = isInView(x, y, myView);
		if(inView){
			if(onDropListener!=null){
				onDropListener.onDrop(dragSrcPosition);
			}
		}
	}
	
	private Bitmap getScaleBitmap(Bitmap bitmap){
		Matrix matrix = new Matrix();
		matrix.postScale(1.1f, 1.1f);
		return Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true);
	}
	
	private boolean isInView(int x, int y, View view){
		if(view == null) return false;
		
		int[] location = new int[2];
		view.getLocationInWindow(location);
		int _x = location[0];
		int _y = location[1];
		int width = view.getWidth();
		int height = view.getHeight();
		
		if(_x <x && x <_x+width && _y<y && y<_y+height){
			return true;
		}
		return false;
	}
	
	

}

相关文章
|
2月前
|
数据可视化 Android开发 开发者
安卓应用开发中的自定义View组件
【10月更文挑战第5天】在安卓应用开发中,自定义View组件是提升用户交互体验的利器。本篇将深入探讨如何从零开始创建自定义View,包括设计理念、实现步骤以及性能优化技巧,帮助开发者打造流畅且富有创意的用户界面。
96 0
|
1月前
|
XML 前端开发 Android开发
Android:UI:Drawable:View/ImageView与Drawable
通过本文的介绍,我们详细探讨了Android中Drawable、View和ImageView的使用方法及其相互关系。Drawable作为图像和图形的抽象表示,提供了丰富的子类和自定义能力,使得开发者能够灵活地实现各种UI效果。View和ImageView则通过使用Drawable实现了各种图像和图形的显示需求。希望本文能为您在Android开发中使用Drawable提供有价值的参考和指导。
40 2
|
1月前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
38 5
|
2月前
|
缓存 数据处理 Android开发
在 Android 中使用 RxJava 更新 View
【10月更文挑战第20天】使用 RxJava 来更新 View 可以提供更优雅、更高效的解决方案。通过合理地运用操作符和订阅机制,我们能够轻松地处理异步数据并在主线程中进行 View 的更新。在实际应用中,需要根据具体情况进行灵活运用,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在 Android 中使用 RxJava 更新 View 的技巧和方法,为开发高质量的 Android 应用提供有力支持。
|
2月前
|
缓存 调度 Android开发
Android 在子线程更新 View
【10月更文挑战第21天】在 Android 开发中,虽然不能直接在子线程更新 View,但通过使用 Handler、AsyncTask 或 RxJava 等方法,可以实现子线程操作并在主线程更新 View 的目的。在实际应用中,需要根据具体情况选择合适的方法,并注意相关的注意事项和性能优化,以确保应用的稳定性和流畅性。可以通过不断的实践和探索,进一步掌握在子线程更新 View 的技巧和方法,为开发高质量的 Android 应用提供支持。
41 2
|
2月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
|
2月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
29 2
|
2月前
|
XML 前端开发 Android开发
Android View的绘制流程和原理详细解说
Android View的绘制流程和原理详细解说
50 3
|
2月前
|
XML 前端开发 Java
安卓应用开发中的自定义View组件
【10月更文挑战第5天】自定义View是安卓应用开发的一块基石,它为开发者提供了无限的可能。通过掌握其原理和实现方法,可以创造出既美观又实用的用户界面。本文将引导你了解自定义View的创建过程,包括绘制技巧、事件处理以及性能优化等关键步骤。
|
3月前
|
缓存 搜索推荐 Android开发
安卓应用开发中的自定义View组件实践
【9月更文挑战第10天】在安卓开发领域,自定义View是提升用户体验和实现界面个性化的重要手段。本文将通过一个实际案例,展示如何在安卓项目中创建和使用自定义View组件,包括设计思路、实现步骤以及可能遇到的问题和解决方案。文章不仅提供了代码示例,还深入探讨了自定义View的性能优化技巧,旨在帮助开发者更好地掌握这一技能。