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;
	}
	
	

}

相关文章
|
3天前
|
缓存 测试技术 Android开发
深入探究Android中的自定义View绘制优化策略
【4月更文挑战第8天】 在Android开发实践中,自定义View的绘制性能至关重要,尤其是当涉及到复杂图形和动画时。本文将探讨几种提高自定义View绘制效率的策略,包括合理使用硬件加速、减少不必要的绘制区域以及利用缓存机制等。这些方法不仅能改善用户体验,还能提升应用的整体性能表现。通过实例分析和性能测试结果,我们将展示如何有效地实现这些优化措施,并为开发者提供实用的技术指南。
|
3天前
|
前端开发 Android开发 开发者
深入探究Android中的自定义View组件开发
【4月更文挑战第3天】 在现代Android应用程序的开发过程中,创建具有独特功能和高度定制化的用户界面是一个常见需求。为此,理解并掌握自定义View组件的开发成为了开发者必备的技能之一。本文将深入探讨如何在Android中创建自定义View,从基础的绘制原理到事件处理机制,再到性能优化技巧,旨在为读者提供一个全面的技术视角,并通过实例代码演示如何实现一个功能丰富、响应迅速的自定义View组件。
|
3天前
|
Android开发 容器
Android UI设计: 什么是View和ViewGroup?
Android UI设计: 什么是View和ViewGroup?
40 0
|
3天前
|
Android开发
Android 分享机顶盒项目的封装类《GridView》(二)(转)
Android 分享机顶盒项目的封装类《GridView》(二)(转)
16 2
|
3天前
|
Android开发
Android通过手势(多点)缩放和拖拽图片
Android通过手势(多点)缩放和拖拽图片
13 4
|
1天前
|
Android开发 异构计算 前端开发
Android显示原理,安卓自定义view面试
Android显示原理,安卓自定义view面试
|
2天前
|
Android开发 开发者 容器
Android自定义View:MeasureSpec的真正意义与View大小控制
Android自定义View:MeasureSpec的真正意义与View大小控制
|
3天前
|
存储 消息中间件 缓存
Android应用开发:实现自定义View的高效绘制
【5月更文挑战第12天】 在Android开发中,创建高性能的自定义视图是一项挑战,它要求开发者深入理解Android的绘图机制以及UI渲染过程。本文将探讨如何优化自定义View的绘制流程,减少不必要的重绘和布局计算,以提升应用的响应速度和流畅度。我们将介绍几种关键策略,包括利用硬件加速、缓存绘制内容和使用高效的数据结构来存储视图状态。通过实例分析和性能对比,读者将学会如何在自己的应用中运用这些技巧,从而打造出更加流畅和响应迅速的用户界面。
|
3天前
|
XML Java Android开发
Android 分享机顶盒项目的封装类《GridView》(三)(转)
Android 分享机顶盒项目的封装类《GridView》(三)(转)
11 2
|
3天前
|
Android开发
Android教程之Android 用户界面-表格视图(GridView)
Android教程之Android 用户界面-表格视图(GridView)
11 1