Android评论图片可移动顺序选择器(推荐)
好久没写了,现在在广州一家公司实习了,来了一个月了,实习生没什么事干,看到公司一个项目。android和ios的做的不一样(ios做这个项目的人多,额不解释。。原来做这个玩意的也跳槽了),既ios的做的控件更酷炫,我闲着没事,把其中的一个控件和ios做的差不多了,来看看效果吧
截的gif图看上去有点快了,因为csdn上传图片不能超过两m所以帧有点大,实际效果是正常的。好了,先让我们看看不能移动交换顺序之前是怎么实现的吧。
package com.test.jiupin.view; import android.content.context; import android.util.attributeset; import android.view.view; import android.widget.framelayout; /** * created by liaoyalong on 2016/12/8. */ public class addimagegridview extends framelayout{ private int width = 0; //图片宽 private int height = 0; //图片高 private int space = dp2px(10); //图片之间间隙 private int childcount = 0; //孩子数 public addimagegridview(context context) { this(context,null); } public addimagegridview(context context, attributeset attrs) { this(context, attrs,0); } public addimagegridview(context context, attributeset attrs, int defstyleattr) { super(context, attrs, defstyleattr); init(); } private void init() { //额没用上 } public int getmspace(){ return space; } public int getmwidth() { return width; } public int getmheight() { //这里我设置了宽高一样 ,所以也没用上 return height; } public void addcammary(view view){ //添加相机 ,是第一个孩子 addview(view,0); } public void addview(view view){ //添加子vie后控件会自动重新测量布局 childcount = getchildcount(); if(childcount == 5){ //最多能添加5张图片,既当添加到第五张图片的时候把相机删了 removeviewat(0); addview(view,4); } else{ addview(view,childcount); } } @override //最关键的了 protected void onmeasure(int widthmeasurespec, int heightmeasurespec) { childcount = getchildcount(); int wdwidth = measurespec.getsize(widthmeasurespec); width = (wdwidth - 3 * space) / 4; //屏幕分为三份间隙 和四份图片的宽度 space += wdwidth % 4 / 3; //重新计算间隙 height = width; //高度和宽度一样 int childwidthspec = measurespec.makemeasurespec(width,measurespec.exactly); //精确测量 int childheightspec = measurespec.makemeasurespec(height,measurespec.exactly); for (int i = 0; i < childcount; i++) { view view = getchildat(i); view.measure(childwidthspec,childheightspec); } int wdheight = height; if(childcount > 4){ wdheight += height + space; //最多有五个既超过四个,高度变成两层 } setmeasureddimension(wdwidth,wdheight); } @override protected void onlayout(boolean changed, int left, int top, int right, int bottom) { for (int i = 0; i < childcount; i++) { //上面我们自己测量好了,这里我们自己放控件的位置,不需要管原来left,top之类的值 view view = getchildat(i); left = 0; top = 0; if(i > 0){ left = i * width + i * space; //每个孩子对应的位置,自己画图分析 } if(i == 4){ //第二层 left = 0; top = height + space; } right = left + width; bottom = top + height; view.layout(left,top,right,bottom); } } public int dp2px(int dp){ return (int) (getresources().getdisplaymetrics().density * dp +.5); } }
才一百行不到,很容易看懂吧,如果对测量不懂的,可以看我以前写的自定义控件,里面的。这就可以了,实现了添加那种不能移动图片的控件。现在开始来做可以移动的吧,我是这样做的。当按图片超过一秒的时候,让这个图片隐藏,然后用windowmanager添加一个可以自定义moveview来显示这个图片的bitmap,并把图片的touch事件也交给添加的moveview来处理,然后通过moveview的移动来判断需不需要与其它图片交换位置.moveview就像我以前的放腾讯拖到小球那样,来看看具体的吧。
package com.test.jiupin.view; import android.content.context; import android.graphics.bitmap; import android.graphics.canvas; import android.graphics.rect; import android.view.motionevent; import android.view.view; /** * created by liaoyalong on 2016/12/12. */ public class moveview extends view{ private bitmap mbitmap; //按1秒钟图片的bitmap 既原显示的都是imageview 我把它们显示的bitmap 用iv.set(bitmap) private int left; //相对屏幕的位置,画的时候要用 private int top; //同上理 private int width; //点击的图片的宽度 private int height; private int mstatusbarheight; //手机屏幕状态栏高度 private int wdwidth; //屏幕宽度 private int wdheight; //屏幕高度 private int orgtop; //刚点击时 点击图片相对屏幕的高度 public moveview(context context,view v,motionevent event,boolean is4,int spac) { super(context); mbitmap = (bitmap) v.gettag(); //从点击的图片那获取bitmap left = (int) (event.getrawx() - event.getx()); //相对屏幕像素 - 相对控件像素 top = (int) (event.getrawy() - event.gety()); orgtop = top; width = v.getwidth(); height = v.getheight(); if (is4){ //第二层 orgtop = orgtop - height - spac; } mstatusbarheight = getstatusbarheight(); //状态栏高度 wdwidth = getresources().getdisplaymetrics().widthpixels; // wdheight = getresources().getdisplaymetrics().heightpixels; } @override protected void ondraw(canvas canvas) { canvas.save(); canvas.translate(0,-mstatusbarheight);//移状态栏高度,这里不懂看我前面qq移动小球的文章 if(left < 0){ left = 0; }else if(left + width > wdwidth){ //不能移出屏幕 left = wdwidth - width; } if (top < mstatusbarheight){ top = mstatusbarheight; }else if(top + height > wdheight){ top = wdheight - height; } canvas.drawbitmap(mbitmap,null,new rect(left,top,left+width,top+height),null); canvas.restore(); } @override public boolean ontouchevent(motionevent event) { switch (event.getaction()){ case motionevent.action_down: break; case motionevent.action_move: left = (int) (event.getrawx()-width/2); //手指点的为中心点 top = (int) (event.getrawy()-height/2); if (mondraglistener != null){ //回调方法,返回中心点与其它图片进行比较 mondraglistener.onmove((int)event.getrawx(),(int)event.getrawy() +mstatusbarheight - orgtop - height/2); } postinvalidate(); break; case motionevent.action_up: if (mondraglistener != null){ //手抬起时触发消失回调 mondraglistener.ondisappear(); } break; } return true; } public int getstatusbarheight() {//获取状态栏高度 int result = 0; int resourceid = getresources().getidentifier("status_bar_height", "dimen", "android"); if (resourceid > 0) { result = getresources().getdimensionpixelsize(resourceid); } return result; } public interface ondraglistener{ void ondisappear(); void onmove(int centerx,int centery); } private ondraglistener mondraglistener; public void setondraglistener(ondraglistener listener){ mondraglistener = listener; } }
恩,注释写的很相许了吧,这里控件搞好了,来activity中看看具体的控制问题吧,首先先看看比较函数
/** * 该不该交换图片 * * @param wx 移动传回来的中心点 * @param wy * @param x 与之相比较的点 * @param y * @param range 比较范围 * @return */ private boolean shouldreplace(int wx, int wy, int x, int y, int range) { boolean flag = false; if (wx > x - range && wx < x + range && wy > y - range && wy < y + range) { flag = true; } return flag; }
确定了比较函数就是,我们还要想改怎么进行比较,我的思路是,前面添加view的既addimagegridview这个控件,它添加一个非相机子view,我就把它存入一个list,然后给list中存的所有view设置touch事件,以一个int 型的currentspace记录当前被长按1秒钟以上的view 在list的位置,并bitmap型的firselect 记录被长按图片的bitmap,然后让它隐藏,当需要交换的时候,假设交换的是j 我把currentspace对应的view设置bitmap为j的bitmap,并让j把当前显示的bitmap settag().再让j对应的view隐藏,并把currentspace设置为j。额。。。很难看懂我在说什么是吧,我自己也觉得好难说清,来看代码吧,我尽量把注释标注详细,等下我也会把完整代码下载链接放在评论区。
private void initmovelistener() { for (int i = 0; i < mviews.size(); i++) { //mviews存的那些评论图片 final int finali = i; mviews.get(i).setontouchlistener(new view.ontouchlistener() { @override public boolean ontouch(final view v, final motionevent event) { if (motionevent.action_down == event.getaction()) { downtime = system.currenttimemillis(); //按下时的时间 v.getparent().requestdisallowintercepttouchevent(true); //请求父控件不要拦截触摸事件 } else { spactime = system.currenttimemillis() - downtime; //按了多久 } // log.e("test",spactime+""); if (!first && spactime > 1000) { //按了一秒才让移动 curentspace = finali; //当前按得位置 first = true; v.setvisibility(view.invisible); //把按得图片隐藏 if (mviews.size() < 5) { //创建移动图片 if (finali != 3) { mmoveview = new moveview(mainactivity.this, v, event, false, 0); } else { mmoveview = new moveview(mainactivity.this, v, event, true, maddimagegridview .getmspace()); } } else { if (finali != 4) { mmoveview = new moveview(mainactivity.this, v, event, false, 0); } else { mmoveview = new moveview(mainactivity.this, v, event, true, maddimagegridview .getmspace()); } } mwindowmanager.addview(mmoveview, mparams);//添加移动图片,注意mparams.format = pixelformat.translucent; fisselect = (bitmap) v.gettag(); //刚开始按选择的图片,用于放手时显示, mmoveview.setondraglistener(new moveview.ondraglistener() {//moveview的滑动监听 @override public void ondisappear() { //手抬起时 if (mmoveview != null && mwindowmanager != null) { mwindowmanager.removeview(mmoveview); //移除 v.setontouchlistener(new view.ontouchlistener() { //消失的时候覆盖触摸事件,不然第二次就不需要按一秒了 @override public boolean ontouch(view v, motionevent event) { return false; } }); mmoveview = null; } // log.e("test","curentspace:"+curentspace); imageview iv = (imageview) mviews.get(curentspace); iv.setimagebitmap(fisselect); //第一次按时选择的图片 iv.setvisibility(view.visible); iv.settag(fisselect); first = false; spactime = 0l; initmovelistener(); } int width = maddimagegridview.getmwidth(); //图片宽度 int spc = maddimagegridview.getmspace(); //图片之间间隙 @override public void onmove(int centerx, int centery) { //返回中心点的回调 for (int j = 0; j < mviews.size(); j++) { //算各个图片的中心点,依次进行比较 if (j != curentspace) { int x = 0; int y = 0; if (mviews.size() < 5) { if (j < 3) { //第一排的中心点 x = (j + 2) * width + (j + 1) * spc - width / 2; y = width / 2; } else if (j != curentspace && j >= 3) { //第二排的中心点。图片宽高相等 x = (j - 3) * (width + spc) + width / 2; y = width + spc + width / 2; } } else { if (j < 4) { //第一排的中心点 x = (j + 1) * width + j * spc - width / 2; y = width / 2; } else if (j != curentspace && j >= 4) { //第二排的中心点。图片宽高相等 x = (j - 4) * (width + spc) + width / 2; y = width + spc + width / 2; } } if (shouldreplace(centerx, centery, x, y, 70)) { //如果要与j位置的view进行交换 // log.e("test","currenspac:"+curentspace + " j:"+j); bitmap bitmap = (bitmap) mviews.get(j).gettag(); //获得j位置的bitmap mviews.get(j).setvisibility(view.invisible); //把j位置对应的view隐藏 imageview iv = (imageview) mviews.get(curentspace); //原来隐藏的bitmap iv.setimagebitmap(bitmap); //原来隐藏的view换成j的bitmap iv.setvisibility(view.visible); //把原来隐藏的view显示 iv.settag(bitmap); //如果交换,把交换的那个隐藏,原来隐藏的显示,并把图片设置为交换的、还要保持tag curentspace = j; //当前空位换为交换既隐藏的那个位置,当松手时要显示,最原始的的bitmap前面已经保存 // log.e("test","currenspac:"+curentspace + " j:"+j); break; //交换停止本次循环 } } } } }); } if (mmoveview != null) { mmoveview.ontouchevent(event); //把触摸事件传递给move控件 } return false; //还要处理点击事件 } }); } }
以上所述是小编给大家介绍的android评论图片可移动顺序选择器,希望对大家有所帮助