google官网的training和API两个地方都提到了拖拽的实现,两种方法不太一样。

方法一

training(https://developer.android.com/training/gestures/scale.html)中提到的方法是监听onTouchEvent,在ACTION_DOWN的时候记录位置,在ACTION_MOVE的时候获取坐标,改变拖拽的控件位置。

方法二

在android3.0及以上可以使用View.OnDragListener。

下面是我写的简单的demo:

activity_drag_demo.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:id="@+id/activity_drag_demo"
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="com.topsports.testapplication.DragDemoActivity">
<ImageView
android:id="@+id/imageView"
android:layout_width="100dp"
android:layout_height="100dp"
android:background="@drawable/c1"/> <FrameLayout
android:background="#c7f6ff"
android:layout_width="500dp"
android:layout_height="500dp"
android:id="@+id/content_fragment"
android:layout_alignParentBottom="true"></FrameLayout> <FrameLayout
android:background="#dce9c1"
android:layout_width="200dp"
android:layout_height="200dp"
android:id="@+id/content_fragment2"
android:layout_alignParentRight="true"
android:layout_alignParentBottom="true"></FrameLayout> </RelativeLayout>
在官网示例基础上修改的DragDemoActivity:package com.topsports.testapplication;
import android.content.ClipData;
import android.content.ClipDescription;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Point;
import android.graphics.drawable.ColorDrawable;
import android.graphics.drawable.Drawable;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.DragEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.Toast; public class DragDemoActivity extends AppCompatActivity { private ImageView imageView; private FrameLayout frameLayout;
private FrameLayout frameLayout2; private static final String IMAGEVIEW_TAG = "icon bitmap"; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_drag_demo); imageView=(ImageView)findViewById(R.id.imageView);
frameLayout=(FrameLayout)findViewById(R.id.content_fragment);
frameLayout2=(FrameLayout)findViewById(R.id.content_fragment2); imageView.setTag(IMAGEVIEW_TAG); imageView.setOnLongClickListener(new View.OnLongClickListener() { public boolean onLongClick(View v) { v.getBackground().setAlpha(100); ClipData.Item item = new ClipData.Item(v.getTag().toString()); ClipData dragData = new ClipData(v.getTag().toString(),new String[]{ClipDescription.MIMETYPE_TEXT_PLAIN},item); // Instantiates the drag shadow builder.
View.DragShadowBuilder myShadow = new View.DragShadowBuilder(v);
v.startDrag(dragData, // the data to be dragged
myShadow, // the drag shadow builder
null, // no need to use local data
0 // flags (not currently used, set to 0)
); return true; }
}); frameLayout.setOnDragListener(new myDragEventListener());
frameLayout2.setOnDragListener(new myDragEventListener()); imageView.setOnDragListener(new myDragEventListener()); }protected class myDragEventListener implements View.OnDragListener { // This is the method that the system calls when it dispatches a drag event to the
// listener.
public boolean onDrag(View v, DragEvent event) { // Defines a variable to store the action type for the incoming event
final int action = event.getAction(); // Handles each of the expected events
switch(action) { case DragEvent.ACTION_DRAG_STARTED: // Determines if this View can accept the dragged data
if (event.getClipDescription().hasMimeType(ClipDescription.MIMETYPE_TEXT_PLAIN)) { v.invalidate(); // returns true to indicate that the View can accept the dragged data.
return true; } // Returns false. During the current drag and drop operation, this View will
// not receive events again until ACTION_DRAG_ENDED is sent.
return false; case DragEvent.ACTION_DRAG_ENTERED: // Applies a green tint to the View. Return true; the return value is ignored // Invalidate the view to force a redraw in the new tint
v.invalidate(); return true; case DragEvent.ACTION_DRAG_LOCATION: // Ignore the event
return true; case DragEvent.ACTION_DRAG_EXITED: // Re-sets the color tint to blue. Returns true; the return value is ignored. // Invalidate the view to force a redraw in the new tint
v.invalidate(); return true; case DragEvent.ACTION_DROP: // Gets the item containing the dragged data
ClipData.Item item = event.getClipData().getItemAt(0); // Gets the text data from the item.
String dragData = item.getText().toString(); // Displays a message containing the dragged data.
Toast.makeText(DragDemoActivity.this, "Dragged data is " + dragData, Toast.LENGTH_LONG).show(); // Turns off any color tints
if(v instanceof FrameLayout){
((ViewGroup)imageView.getParent()).removeView(imageView);
((FrameLayout)v).addView(imageView);
}
// Invalidates the view to force a redraw
v.invalidate(); // Returns true. DragEvent.getResult() will return true.
return true; case DragEvent.ACTION_DRAG_ENDED: // Turns off any color tinting // Invalidates the view to force a redraw
v.invalidate(); // Does a getResult(), and displays what happened.
if (event.getResult()) {
Toast.makeText(DragDemoActivity.this, "The drop was handled.", Toast.LENGTH_LONG).show(); } else {
Toast.makeText(DragDemoActivity.this, "The drop didn't work.", Toast.LENGTH_LONG).show(); }
imageView.getBackground().setAlpha(255);
// returns true; the value is ignored.
return true; // An unknown action type was received.
default:
Log.e("DragDrop Example","Unknown action type received by OnDragListener.");
break;
} return false;
}
}
}

实现的是将图片拖进两个layout中,并在两个layout间来回拖动。

代码做简单的解释

在长按imageview的时候开始拖拽事件

ClipData.Item item = new ClipData.Item(v.getTag().toString());

ClipData dragData = new ClipData(v.getTag().toString(),new String[]{ClipDescription.MIMETYPE_TEXT_PLAIN},item);

这两句是为拖动的view创建传递的数据(可以省略)

View.DragShadowBuilder myShadow = new View.DragShadowBuilder(v)

这句根据imageview(v)创建一个拖拽的阴影,就像在桌面上拖动应用图标一样,产生一个view的图像副本,产生拖动效果。

frameLayout.setOnDragListener(new myDragEventListener());

frameLayout2.setOnDragListener(new myDragEventListener());

imageView.setOnDragListener(new myDragEventListener());

设置拖拽的事件监听(拖动的对象和被放置的对象)实现View.OnDragListener的onDrag方法,判断event.getAction()的值进行不同的操作。

 

Android开发:拖拽的更多相关文章

  1. Android 可拖拽的GridView效果实现, 长按可拖拽和item实时交换

    转帖请注明本文出自xiaanming的博客(http://blog.csdn.net/xiaanming/article/details/17718579),请尊重他人的辛勤劳动成果,谢谢! 在And ...

  2. Android Launcher拖拽事件详解【android4.0--Launcher系列二】

    AndroidICS4.0版本的launcher拖 拽的流程,基本和2.3的相似.就是比2.3写的封装的接口多了一些,比如删除类的写法就多了个类.等等.4.0的改变有一些,但是不是特别大.这个月一 直 ...

  3. android五子棋游戏、资讯阅读、大学课程表、地图拖拽检测、小说搜索阅读app等源码

    Android精选源码 Android 自动生成添加控件 android旋转动画.圆形进度条组合效果源码 一款很强的手机五子棋app源码 android地图拖拽区域检测效果源码 实现Android大学 ...

  4. Java实现拖拽上传

    原文:http://www.open-open.com/code/view/1437358795584 在项目开发中由于实际需求,需要开发拖拽上传的功能,ok! 先看效果图: jsp上传前端代码: & ...

  5. android开发游记:SpringView 下拉刷新的高效解决方式,定制你自己风格的拖拽页面

    关于下拉刷新/上拉载入很多其它的解决方式网上已经有非常多了,浏览了眼下主流的下拉控件比方PullToRefresh库等.第一:大多数实现库都难以进行动画和样式的自己定义. 第二:不能非常好的兼容多种滚 ...

  6. android开发游记:ItemTouchHelper 使用RecyclerView打造可拖拽的GridView

    以下是RecyclerView结合ItemTouchHelper实现的列表和网格布局的拖拽效果. 效果图例如以下:(gif图有点顿卡,事实上执行是非常流畅的) demo下载地址: DragRecycl ...

  7. Android中GridView拖拽的效果【android进化三十六】

      最 近看到联想,摩托罗拉等,手机launcher中有个效果,进入mainmenu后,里面的应用程序的图标可以拖来拖去,所以我也参照网上给的代码,写了 一个例子.还是很有趣的,实现的流畅度没有人家的 ...

  8. Android中GridView拖拽的效果

    最 近看到联想,摩托罗拉等,手机launcher中有个效果,进入mainmenu后,里面的应用程序的图标可以拖来拖去,所以我也参照网上给的代码,写了 一个例子.还是很有趣的,实现的流畅度没有人家的那么 ...

  9. android银行卡匹配、详情展开动画、仿爱奇艺视频拖拽、扫码识别手机号等源码

    Android精选源码 android实现银行卡匹配信息源码 android实现可以展开查看详情的卡片 下拉刷新,上拉加载,侧滑显示菜单等效果RefreshSwipeRecyclerview andr ...

随机推荐

  1. N的阶层(王道)

    题目描述: 输入一个正整数N,输出N的阶乘. 输入: 正整数N(0<=N<=1000) 输出: 输入可能包括多组数据,对于每一组输入数据,输出N的阶乘 样例输入: 4 5 15 样例输出: ...

  2. Linux(CentOS)修改IP地址

    登陆连接centos系统,输入 ifconfig 可以查看到当前本机的IP地址信息 一 临时修改IP地址: 1.假如查询IP为1.118,输入 ifconfig eth0 (默认是第一个网卡) 后面接 ...

  3. win7安装centos7双系统

    采用硬盘安装 前景 打算用U盘安装,但是u盘是FAT32格式限制了文件4g大小,我官网下的iso镜像大于4g,只好采用硬盘安装. 其实U盘安装是最方便的,网上很多教程用UltraISO软件把U盘直接作 ...

  4. 【Qt编程】基于Qt的词典开发系列&lt;八&gt;--用户登录及API调用的实现

    在上一篇文章<调用网络API>中,我仅仅讲述了怎样直观的使用API接口以及调用API后返回的结果,本文则从程序实现的角度来实现API的调用.当然本程序的实现也是借助于扇贝网的API接口文档 ...

  5. Android之——获取手机安装的应用程序

    转载请注明出处:http://blog.csdn.net/l1028386804/article/details/47114331 前几篇有关Android的博文中.向大家介绍了几个项目中经常使用的有 ...

  6. 开源的C# websocket-sharp组件解析

    下面我们介绍一款WebSocket组件websocket-sharp的相关内容. 一.websocket-sharp组件概述 websocket-sharp是一个C#实现websocket协议客户端和 ...

  7. Python——学好Python必读的几篇文章

    作为脚本语言Python上手容易,但要学好Python能写出一手漂亮的.Pythonic的Python代码并非一日之功,本文的目的在于推荐 一些优秀的Python相关的文章(至于书大家可以看dip.l ...

  8. windows 7 提示升级到windows 10补丁

    如果不需要这个提示,可以卸载KB3035583和KB2952664这两个系统更新补丁.   other update:KB2976978   and  KB2977759

  9. ubuntu cp(copy) command

    e.g. #cp –p –R  /opt/lampp/drv ~/test   Ubuntu_Linux命令:cp(copy)复制文件或目录 功能: 复制文件或目录   www.2cto.com  说 ...

  10. excel单元格内插入选择项pass、fail、not support等

    1.点击菜单栏的数据—-->>数据验证 2.选择 序列 在 来源 选项中填入Pass,Fail,On Going,Not Support 3.在选中的单元格并在菜单栏选中 新建规则