Android評(píng)論圖片可移動(dòng)順序選擇器(推薦)
好久沒(méi)寫(xiě)了,現(xiàn)在在廣州一家公司實(shí)習(xí)了,來(lái)了一個(gè)月了,實(shí)習(xí)生沒(méi)什么事干,看到公司一個(gè)項(xiàng)目。Android和iOS的做的不一樣(ios做這個(gè)項(xiàng)目的人多,額不解釋。。原來(lái)做這個(gè)玩意的也跳槽了),既ios的做的控件更酷炫,我閑著沒(méi)事,把其中的一個(gè)控件和IOS做的差不多了,來(lái)看看效果吧
截的GIF圖看上去有點(diǎn)快了,因?yàn)镃SDN上傳圖片不能超過(guò)兩M所以幀有點(diǎn)大,實(shí)際效果是正常的。好了,先讓我們看看不能移動(dòng)交換順序之前是怎么實(shí)現(xiàn)的吧。
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; //孩子數(shù) 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() { //額沒(méi)用上 } public int getmSpace(){ return space; } public int getmWidth() { return width; } public int getmHeight() { //這里我設(shè)置了寬高一樣 ,所以也沒(méi)用上 return height; } public void addCammary(View view){ //添加相機(jī) ,是第一個(gè)孩子 addView(view,0); } public void addView(View view){ //添加子vie后控件會(huì)自動(dòng)重新測(cè)量布局 childCount = getChildCount(); if(childCount == 5){ //最多能添加5張圖片,既當(dāng)添加到第五張圖片的時(shí)候把相機(jī)刪了 removeViewAt(0); addView(view,4); } else{ addView(view,childCount); } } @Override //最關(guān)鍵的了 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { childCount = getChildCount(); int wdWidth = MeasureSpec.getSize(widthMeasureSpec); width = (wdWidth - 3 * space) / 4; //屏幕分為三份間隙 和四份圖片的寬度 space += wdWidth % 4 / 3; //重新計(jì)算間隙 height = width; //高度和寬度一樣 int childWidthSPEC = MeasureSpec.makeMeasureSpec(width,MeasureSpec.EXACTLY); //精確測(cè)量 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; //最多有五個(gè)既超過(guò)四個(gè),高度變成兩層 } setMeasuredDimension(wdWidth,wdHeight); } @Override protected void onLayout(boolean changed, int left, int top, int right, int bottom) { for (int i = 0; i < childCount; i++) { //上面我們自己測(cè)量好了,這里我們自己放控件的位置,不需要管原來(lái)left,top之類(lèi)的值 View view = getChildAt(i); left = 0; top = 0; if(i > 0){ left = i * width + i * space; //每個(gè)孩子對(duì)應(yīng)的位置,自己畫(huà)圖分析 } 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); } }
才一百行不到,很容易看懂吧,如果對(duì)測(cè)量不懂的,可以看我以前寫(xiě)的自定義控件,里面的。這就可以了,實(shí)現(xiàn)了添加那種不能移動(dòng)圖片的控件?,F(xiàn)在開(kāi)始來(lái)做可以移動(dòng)的吧,我是這樣做的。當(dāng)按圖片超過(guò)一秒的時(shí)候,讓這個(gè)圖片隱藏,然后用WindowManager添加一個(gè)可以自定義moveView來(lái)顯示這個(gè)圖片的BitMap,并把圖片的touch事件也交給添加的moveView來(lái)處理,然后通過(guò)moveView的移動(dòng)來(lái)判斷需不需要與其它圖片交換位置.moveView就像我以前的放騰訊拖到小球那樣,來(lái)看看具體的吧。
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; //相對(duì)屏幕的位置,畫(huà)的時(shí)候要用 private int top; //同上理 private int width; //點(diǎn)擊的圖片的寬度 private int height; private int mStatusBarheight; //手機(jī)屏幕狀態(tài)欄高度 private int wdWidth; //屏幕寬度 private int wdHeight; //屏幕高度 private int orgTop; //剛點(diǎn)擊時(shí) 點(diǎn)擊圖片相對(duì)屏幕的高度 public moveView(Context context,View v,MotionEvent event,boolean is4,int spac) { super(context); mBitmap = (Bitmap) v.getTag(); //從點(diǎn)擊的圖片那獲取bitmap left = (int) (event.getRawX() - event.getX()); //相對(duì)屏幕像素 - 相對(duì)控件像素 top = (int) (event.getRawY() - event.getY()); orgTop = top; width = v.getWidth(); height = v.getHeight(); if (is4){ //第二層 orgTop = orgTop - height - spac; } mStatusBarheight = getStatusBarHeight(); //狀態(tài)欄高度 wdWidth = getResources().getDisplayMetrics().widthPixels; // wdHeight = getResources().getDisplayMetrics().heightPixels; } @Override protected void onDraw(Canvas canvas) { canvas.save(); canvas.translate(0,-mStatusBarheight);//移狀態(tài)欄高度,這里不懂看我前面QQ移動(dòng)小球的文章 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); //手指點(diǎn)的為中心點(diǎn) top = (int) (event.getRawY()-height/2); if (mOnDragListener != null){ //回調(diào)方法,返回中心點(diǎn)與其它圖片進(jìn)行比較 mOnDragListener.onMove((int)event.getRawX(),(int)event.getRawY() +mStatusBarheight - orgTop - height/2); } postInvalidate(); break; case MotionEvent.ACTION_UP: if (mOnDragListener != null){ //手抬起時(shí)觸發(fā)消失回調(diào) mOnDragListener.onDisappear(); } break; } return true; } public int getStatusBarHeight() {//獲取狀態(tài)欄高度 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; } }
恩,注釋寫(xiě)的很相許了吧,這里控件搞好了,來(lái)activity中看看具體的控制問(wèn)題吧,首先先看看比較函數(shù)
/** * 該不該交換圖片 * * @param wx 移動(dòng)傳回來(lái)的中心點(diǎn) * @param wy * @param x 與之相比較的點(diǎn) * @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; }
確定了比較函數(shù)就是,我們還要想改怎么進(jìn)行比較,我的思路是,前面添加view的既AddImageGridView這個(gè)控件,它添加一個(gè)非相機(jī)子view,我就把它存入一個(gè)List,然后給list中存的所有view設(shè)置touch事件,以一個(gè)int 型的CurrentSpace記錄當(dāng)前被長(zhǎng)按1秒鐘以上的view 在list的位置,并Bitmap型的firSelect 記錄被長(zhǎng)按圖片的bitmap,然后讓它隱藏,當(dāng)需要交換的時(shí)候,假設(shè)交換的是j 我把CurrentSpace對(duì)應(yīng)的view設(shè)置bitmap為j的bitmap,并讓j把當(dāng)前顯示的bitmap setTag().再讓j對(duì)應(yīng)的view隱藏,并把CurrentSpace設(shè)置為j。額。。。很難看懂我在說(shuō)什么是吧,我自己也覺(jué)得好難說(shuō)清,來(lái)看代碼吧,我盡量把注釋標(biāo)注詳細(xì),等下我也會(huì)把完整代碼下載鏈接放在評(píng)論區(qū)。
private void initMoveListener() { for (int i = 0; i < mViews.size(); i++) { //mViews存的那些評(píng)論圖片 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(); //按下時(shí)的時(shí)間 v.getParent().requestDisallowInterceptTouchEvent(true); //請(qǐng)求父控件不要攔截觸摸事件 } else { spacTime = System.currentTimeMillis() - downTime; //按了多久 } // Log.e("test",spacTime+""); if (!first && spacTime > 1000) { //按了一秒才讓移動(dòng) curentSpace = finalI; //當(dāng)前按得位置 first = true; v.setVisibility(View.INVISIBLE); //把按得圖片隱藏 if (mViews.size() < 5) { //創(chuàng)建移動(dòng)圖片 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);//添加移動(dòng)圖片,注意mParams.format = PixelFormat.TRANSLUCENT; fisselect = (Bitmap) v.getTag(); //剛開(kāi)始按選擇的圖片,用于放手時(shí)顯示, mMoveView.setOnDragListener(new moveView.OnDragListener() {//moveView的滑動(dòng)監(jiān)聽(tīng) @Override public void onDisappear() { //手抬起時(shí) if (mMoveView != null && mWindowManager != null) { mWindowManager.removeView(mMoveView); //移除 v.setOnTouchListener(new View.OnTouchListener() { //消失的時(shí)候覆蓋觸摸事件,不然第二次就不需要按一秒了 @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); //第一次按時(shí)選擇的圖片 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) { //返回中心點(diǎn)的回調(diào) for (int j = 0; j < mViews.size(); j++) { //算各個(gè)圖片的中心點(diǎn),依次進(jìn)行比較 if (j != curentSpace) { int x = 0; int y = 0; if (mViews.size() < 5) { if (j < 3) { //第一排的中心點(diǎn) x = (j + 2) * width + (j + 1) * spc - width / 2; y = width / 2; } else if (j != curentSpace && j >= 3) { //第二排的中心點(diǎn)。圖片寬高相等 x = (j - 3) * (width + spc) + width / 2; y = width + spc + width / 2; } } else { if (j < 4) { //第一排的中心點(diǎn) x = (j + 1) * width + j * spc - width / 2; y = width / 2; } else if (j != curentSpace && j >= 4) { //第二排的中心點(diǎn)。圖片寬高相等 x = (j - 4) * (width + spc) + width / 2; y = width + spc + width / 2; } } if (shouldReplace(centerX, centerY, x, y, 70)) { //如果要與J位置的view進(jìn)行交換 // Log.e("test","currenspac:"+curentSpace + " j:"+j); Bitmap bitmap = (Bitmap) mViews.get(j).getTag(); //獲得j位置的bitmap mViews.get(j).setVisibility(View.INVISIBLE); //把j位置對(duì)應(yīng)的view隱藏 ImageView iv = (ImageView) mViews.get(curentSpace); //原來(lái)隱藏的bitmap iv.setImageBitmap(bitmap); //原來(lái)隱藏的view換成j的bitmap iv.setVisibility(View.VISIBLE); //把原來(lái)隱藏的view顯示 iv.setTag(bitmap); //如果交換,把交換的那個(gè)隱藏,原來(lái)隱藏的顯示,并把圖片設(shè)置為交換的、還要保持tag curentSpace = j; //當(dāng)前空位換為交換既隱藏的那個(gè)位置,當(dāng)松手時(shí)要顯示,最原始的的bitmap前面已經(jīng)保存 // Log.e("test","currenspac:"+curentSpace + " j:"+j); break; //交換停止本次循環(huán) } } } } }); } if (mMoveView != null) { mMoveView.onTouchEvent(event); //把觸摸事件傳遞給move控件 } return false; //還要處理點(diǎn)擊事件 } }); } }
以上所述是小編給大家介紹的Android評(píng)論圖片可移動(dòng)順序選擇器,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- Android 中NumberPicker,DatePicker與DatePickerDialog中分割顏色的修改實(shí)例代碼
- Android CalendarView,DatePicker,TimePicker,以及NumberPicker的使用
- Android中顏色選擇器和改變字體顏色的實(shí)例教程
- Android PickerView滾動(dòng)選擇器的使用方法
- Android文件選擇器ExFilePicker的使用方法
- Android自定義滾動(dòng)選擇器實(shí)例代碼
- Android自定義可循環(huán)的滾動(dòng)選擇器CycleWheelView
- Android仿iphone自定義滾動(dòng)選擇器
- Android日期選擇器實(shí)現(xiàn)年月日三級(jí)聯(lián)動(dòng)
- Android自定義wheelview實(shí)現(xiàn)滾動(dòng)日期選擇器
- Android開(kāi)發(fā)之滑動(dòng)數(shù)值選擇器NumberPicker用法示例
相關(guān)文章
Android購(gòu)物分類(lèi)效果實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Android購(gòu)物分類(lèi)效果的實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-11-11Android中Volley框架進(jìn)行請(qǐng)求網(wǎng)絡(luò)數(shù)據(jù)的使用
這篇文章主要介紹了Android中Volley框架進(jìn)行請(qǐng)求網(wǎng)絡(luò)數(shù)據(jù)的使用,本文給大家介紹的非常詳細(xì)具有參考借鑒價(jià)值,需要的朋友可以參考下2016-10-10android studio實(shí)現(xiàn)計(jì)算器
這篇文章主要為大家詳細(xì)介紹了android studio實(shí)現(xiàn)計(jì)算器的具體方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07Android 設(shè)置主題實(shí)現(xiàn)點(diǎn)擊波紋效果的示例
本篇文章主要介紹了Android 設(shè)置主題實(shí)現(xiàn)點(diǎn)擊波紋效果的示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11Android編程UI設(shè)計(jì)之GridView和ImageView的用法
這篇文章主要介紹了Android編程UI設(shè)計(jì)之GridView和ImageView的用法,結(jié)合實(shí)例形式較為詳細(xì)的分析了Android中GridView和ImageView組件的相關(guān)方法使用技巧,需要的朋友可以參考下2016-01-01Flutter通過(guò)Container實(shí)現(xiàn)時(shí)間軸效果
時(shí)間軸是前端UI經(jīng)常用到的效果,本文講解下Flutter如何通過(guò)Container實(shí)現(xiàn),感興趣的朋友可以了解下2021-05-05Android實(shí)現(xiàn)手勢(shì)滑動(dòng)和簡(jiǎn)單動(dòng)畫(huà)效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)手勢(shì)滑動(dòng)和簡(jiǎn)單動(dòng)畫(huà)效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05VideoView實(shí)現(xiàn)視頻無(wú)縫連續(xù)播放
這篇文章主要為大家詳細(xì)介紹了VideoView實(shí)現(xiàn)視頻無(wú)縫連續(xù)播放,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07Android TelephonyManager詳解及實(shí)現(xiàn)代碼
本文主要介紹Android TelephonyManager, 這里整理了關(guān)于Android TelephoneManager的相關(guān)資料,并附有示例代碼和實(shí)現(xiàn)效果圖,有需要的朋友可以參考下2016-08-08