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

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

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

@H_404_5@

@H_419_0@截的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,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);  }}
@H_419_0@才一百行不到,很容易看懂吧,如果对测量不懂的,可以看我以前写的自定义控件,里面的。这就可以了,实现了添加那种不能移动图片的控件。现在开始来做可以移动的吧,我是这样做的。当按图片超过一秒的时候,让这个图片隐藏,然后用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,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;  }}
@H_419_0@恩,注释写的很相许了吧,这里控件搞好了,来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;  }
@H_419_0@确定了比较函数就是,我们还要想改怎么进行比较,我的思路是,前面添加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,true,mAddImageGrIDVIEw                    .getmSpace());              }            } else {              if (finali != 4) {                mMoveVIEw = new moveVIEw(MainActivity.this,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; //还要处理点击事件        }      });    }  }
@H_419_0@以上所述是小编给大家介绍的AndroID评论图片可移动顺序选择器,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对编程小技巧网站的支持!

总结

以上是内存溢出为你收集整理的Android评论图片可移动顺序选择器(推荐)全部内容,希望文章能够帮你解决Android评论图片可移动顺序选择器(推荐)所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1147519.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-31
下一篇 2022-05-31

发表评论

登录后才能评论

评论列表(0条)

保存