欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android评论图片可移动顺序选择器(推荐)

程序员文章站 2024-02-28 09:55:58
好久没写了,现在在广州一家公司实习了,来了一个月了,实习生没什么事干,看到公司一个项目。android和ios的做的不一样(ios做这个项目的人多,额不解释。。原来做这个玩...

好久没写了,现在在广州一家公司实习了,来了一个月了,实习生没什么事干,看到公司一个项目。android和ios的做的不一样(ios做这个项目的人多,额不解释。。原来做这个玩意的也跳槽了),既ios的做的控件更酷炫,我闲着没事,把其中的一个控件和ios做的差不多了,来看看效果吧

Android评论图片可移动顺序选择器(推荐)Android评论图片可移动顺序选择器(推荐)

截的gif图看上去有点快了,因为csdn上传图片不能超过两m所以帧有点大,实际效果是正常的。好了,先让我们看看不能移动交换顺序之前是怎么实现的吧。

Android评论图片可移动顺序选择器(推荐)

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评论图片可移动顺序选择器,希望对大家有所帮助