android实现文件的拖拽效果

来源:互联网 发布:linux 开机自启动脚本 编辑:程序博客网 时间:2024/05/26 02:55

在Android 3.0 以前的版本,拖放一个试图需要使用触摸(Touch)事件,而且拖动到指定的区域还需要判断坐标是否落到这一区域,很麻烦。从Android 3.0以后,Android SDK直接支持视图的拖放操作。
拖放操作需要经历的4种状态。

开始拖动
通过调用View.startDrag方法,可以让视图处于可拖动的状态,这时用户可以用手指(虚拟机当然是鼠标啦)将视图在屏幕上拖动。在视图开始拖动的时候,还会使用一种拖动阴影(Drag Shadow) 技术(View.DragShadowBuilder对象),以在Android 3.0 以前的版本,拖放一个试图需要使用触摸(Touch)事件,而且拖动到指定的区域还需要判断坐标是否落到这一区域,很麻烦。从Android 3.0以后,Android SDK直接支持视图的拖放操作。
拖放操作需要经历的4种状态。
开始拖动
通过调用View.startDrag方法,可以让视图处于可拖动的状态,这时用户可以用手指(虚拟机当然是鼠标啦)将视图在屏幕上拖动。在视图开始拖动的时候,还会使用一种拖动阴影(Drag Shadow) 技术(View.DragShadowBuilder对象),以便使拖动的图像与原图形不同。
OnDragListener.onDrag 就是处理拖动操作的方法。方法原型:
public boolean onDrag(View view,DragEvent event);
其中view参数表示当前拖动的视图,event表示拖动过程中的各种信息,其中DragEvent.getAction方法最重要,该方法可以获取具体的拖动状态,如果处于拖动状态,则返回DragEvent.ACTION_DRAG_STARTED。
onDrag方法必须返回true ,如果false,表示当前视图不能拖动。

拖动进行时
这个状态是动态的,当视图进入目标区域,目标区域的onDrag方法就会被调用,这时DragEvent.getAction方法会返回DragEvent.ACTION_DRAG_ENTERED。

放下视图
一旦视图在目标区域放下,这时DragEvent.getAction方法会返回DragEvent.ACTION_DROP,表示视图已经放在了目标区域的某个位置,然后就需要在onDrag方法中做进一步的处理。

结束拖放
当视图放下后(用户松开了手指,拖动阴影消失),不管视图放在了目标区域外,还是目标区域内,系统都会向所有可以监听拖放动作的视图发送DragEvent.ACTION_DRAG_ENDED动作。如果处于目标区域内,发送ACTION_DRAG_ENDED之前,系统会单独向该目标区域内发送ACTION_DROP动作。也就是说只要用户松开正在拖动的视图,DragEvent.getAction方法就一定会返回DragEvent.ACTION_DRAG_ENDED.因此,目标区域接收到DragEvent.ACTION_DRAG_ENDED时不一定是视图放到了目标区域,很可能在目标区域外,所以处理放下动作时要使用DragEvent.ACTON_DROP ,而不要使用DragEvent.ACTION_DRAG_ENDED。便使拖动的图像与原图形不同。
OnDragListener.onDrag 就是处理拖动操作的方法。方法原型:
public boolean onDrag(View view,DragEvent event);
其中view参数表示当前拖动的视图,event表示拖动过程中的各种信息,其中DragEvent.getAction方法最重要,该方法可以获取具体的拖动状态,如果处于拖动状态,则返回DragEvent.ACTION_DRAG_STARTED。
onDrag方法必须返回true ,如果false,表示当前视图不能拖动。

拖动进行时
这个状态是动态的,当视图进入目标区域,目标区域的onDrag方法就会被调用,这时DragEvent.getAction方法会返回DragEvent.ACTION_DRAG_ENTERED。

放下视图
一旦视图在目标区域放下,这时DragEvent.getAction方法会返回DragEvent.ACTION_DROP,表示视图已经放在了目标区域的某个位置,然后就需要在onDrag方法中做进一步的处理。

结束拖放
当视图放下后(用户松开了手指,拖动阴影消失),不管视图放在了目标区域外,还是目标区域内,系统都会向所有可以监听拖放动作的视图发送DragEvent.ACTION_DRAG_ENDED动作。如果处于目标区域内,发送ACTION_DRAG_ENDED之前,系统会单独向该目标区域内发送ACTION_DROP动作。也就是说只要用户松开正在拖动的视图,DragEvent.getAction方法就一定会返回DragEvent.ACTION_DRAG_ENDED.因此,目标区域接收到DragEvent.ACTION_DRAG_ENDED时不一定是视图放到了目标区域,很可能在目标区域外,所以处理放下动作时要使用DragEvent.ACTON_DROP ,而不要使用DragEvent.ACTION_DRAG_ENDED。


拖动阴影
拖动阴影直接使用View.DragShadowBuilder 类 也可以继承View.DragShadowBuilder 类,实现自定义的拖动阴影类。直接使用则拖动的样式与原图像完全一样,只是左上角偏移了一点。如果要自定义阴影类,一般只需要实现View.DragShadowBuilder类的俩个方法:onProvideShadowMetrics和onDragShow。前者用于生成拖动阴影图像(Bitmap对象),后者用于在画布(Canvas)上画出拖动的阴影图像.

不多说了,还是上代码吧。

public class MyDragShadowBuilder extends DragShadowBuilder {// 拖动阴影的区域private static Drawable shadow;// 储存绘制的拖动阴影图像private static Bitmap newBitmap;public MyDragShadowBuilder(View arg0) {super(arg0);// TODO Auto-generated constructor stubshadow = new ColorDrawable(Color.LTGRAY);// 浅灰色}// 在该方法中绘制拖动阴影图像 实例化newBitmap变量@Overridepublic void onProvideShadowMetrics(Point shadowSize, Point shadowTouchPoint) {// TODO Auto-generated method stubsuper.onProvideShadowMetrics(shadowSize, shadowTouchPoint);int width, heigth;// 设置拖动阴影的宽度/高度为原宽/高度的1.5倍width = (int) (getView().getWidth() * 1.5);heigth = (int) (getView().getHeight() * 1.5);// 设置拖动图像的绘制 区域shadow.setBounds(0, 0, width, heigth);// 设置拖动阴影图像的宽度和高度shadowSize.set(width, heigth);// 设置手指在拖动图像的位置 设置为中点shadowTouchPoint.set(width / 2, heigth / 2);if (getView() instanceof ImageView) {// getView()方法返回的值就是构造方法传入的arg0 参数ImageView imageView = (ImageView) getView();// 获取drawable对象Drawable drawable = imageView.getDrawable();// 获取imageview的bitmapBitmap bitmap = ((BitmapDrawable) drawable).getBitmap();// 创建一个新的bitmapnewBitmap = bitmap.createBitmap(width, heigth, Config.ARGB_8888);Canvas canvas = new Canvas(newBitmap);// 将图像绘制在画布上,但现在还没有正式将图像绘制在阴影图像上,目前只是将bitmap放大并绘制在newbitmap上canvas.drawBitmap(newBitmap, new Rect(0, 0, bitmap.getWidth(),bitmap.getHeight()), new Rect(0, 0, width, heigth), null);}}@Overridepublic void onDrawShadow(Canvas canvas) {// TODO Auto-generated method stubsuper.onDrawShadow(canvas);// 将图像正式绘制在阴影图像上canvas.drawBitmap(newBitmap, 0, 0, new Paint());}}



用来实现的主窗口

package com.fang.androidtest;import android.app.Activity;import android.os.Bundle;import android.util.Log;import android.view.DragEvent;import android.view.Menu;import android.view.View;import android.view.View.DragShadowBuilder;import android.view.View.OnDragListener;import android.view.View.OnLongClickListener;import android.widget.ImageView;import android.widget.RelativeLayout;import android.widget.RelativeLayout.LayoutParams;import android.widget.Toast;public class MainActivity extends Activity implements OnDragListener {//把图片拖拽到哪里(可以拖拽到的区域)private RelativeLayout dragdropRegin;// 拖动到图像private ImageView imageView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);

dragdropRegin = (RelativeLayout) findViewById(R.id.fl_dragdrop_region);imageView = (ImageView) findViewById(R.id.imageview);

//给可以拖拽到的区域添加Drag监听器dragdropRegin.setOnDragListener(this);// 为目标设置拖动监听器imageView.setOnLongClickListener(new OnLongClickListener() {@Overridepublic boolean onLongClick(View v) {// TODO Auto-generated method stubDragShadowBuilder mysBuilder = new MyDragShadowBuilder(imageView);// 开始拖动,方法中第一参数是ClipData类型的对象。用于传递剪切板数据,可以为nullv.startDrag(null, mysBuilder, null, 0);return true;}});}@Overridepublic boolean onCreateOptionsMenu(Menu menu) {// Inflate the menu; this adds items to the action bar if it is present.getMenuInflater().inflate(R.menu.main, menu);return true;}@Overridepublic boolean onDrag(View v, DragEvent event) {// TODO Auto-generated method stubint action = event.getAction();switch (action) {// 开始拖动case DragEvent.ACTION_DRAG_STARTED:Toast.makeText(MainActivity.this, "开始拖动", Toast.LENGTH_LONG).show();break;// 进入目标区域case DragEvent.ACTION_DRAG_ENTERED:Toast.makeText(MainActivity.this, "进入目标区域", Toast.LENGTH_LONG).show();break;// 在目标区域移动case DragEvent.ACTION_DRAG_LOCATION:Log.e("aa", "drag location x=" + event.getX() + " y ="+ event.getY());break;// 离开目标区域case DragEvent.ACTION_DRAG_EXITED:Toast.makeText(MainActivity.this, "离开目标区域", Toast.LENGTH_LONG).show();break;// 在目标区域放下ImageView控件case DragEvent.ACTION_DROP:Toast.makeText(MainActivity.this, "在目标区域放下ImageView控件", Toast.LENGTH_LONG).show();ImageView imageView = (ImageView) getLayoutInflater().inflate(R.layout.image, null);LayoutParams layoutParams = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);layoutParams.leftMargin = (int) event.getX()-100;layoutParams.topMargin = (int) event.getY()-100;// 添加到视图中,完成复制dragdropRegin.addView(imageView, layoutParams);break;case DragEvent.ACTION_DRAG_ENDED:Toast.makeText(MainActivity.this, "完成拖拽", Toast.LENGTH_LONG).show();default:return false;}return true;}}


以及布局文件activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:paddingBottom="@dimen/activity_vertical_margin"    android:paddingLeft="@dimen/activity_horizontal_margin"    android:paddingRight="@dimen/activity_horizontal_margin"    android:paddingTop="@dimen/activity_vertical_margin"    tools:context=".MainActivity" >    <RelativeLayout        android:id="@+id/fl_dragdrop_region"        android:layout_width="match_parent"        android:layout_height="match_parent"        android:layout_alignParentTop="true"        android:layout_centerHorizontal="true"        android:orientation="vertical" >        <ImageView            android:id="@+id/imageview"            android:layout_width="wrap_content"            android:layout_height="wrap_content"            android:layout_gravity="center"            android:src="@drawable/ic_launcher" />    </RelativeLayout></RelativeLayout>


image.xml

<?xml version="1.0" encoding="utf-8"?>  <ImageView xmlns:android="http://schemas.android.com/apk/res/android"      android:layout_width="wrap_content"      android:layout_height="wrap_content"      android:src="@drawable/ic_launcher" />  

0 0
原创粉丝点击