Android拖拽詳解

Android中實現拖拽其實很簡單,系統早已經提供了api讓我使用,主要用到了View的startDrag(startDragAndDrop API24+) 方法以及OnDragListenerjava

startDrag

先來看下方法介紹:android

/**
     * Starts a drag and drop operation. When your application calls this method, it passes a
     * {@link android.view.View.DragShadowBuilder} object to the system. The
     * system calls this object's {@link DragShadowBuilder#onProvideShadowMetrics(Point, Point)} * to get metrics for the drag shadow, and then calls the object's
     * {@link DragShadowBuilder#onDrawShadow(Canvas)} to draw the drag shadow itself.
     * <p>
     *  Once the system has the drag shadow, it begins the drag and drop operation by sending
     *  drag events to all the View objects in your application that are currently visible. It does
     *  this either by calling the View object's drag listener (an implementation of * {@link android.view.View.OnDragListener#onDrag(View,DragEvent) onDrag()} or by calling the * View object's {@link android.view.View#onDragEvent(DragEvent) onDragEvent()} method.
     *  Both are passed a {@link android.view.DragEvent} object that has a
     *  {@link android.view.DragEvent#getAction()} value of
     *  {@link android.view.DragEvent#ACTION_DRAG_STARTED}.
     * </p>
     * <p>
     * Your application can invoke {@link #startDragAndDrop(ClipData, DragShadowBuilder, Object,
     * int) startDragAndDrop()} on any attached View object. The View object does not need to be
     * the one used in {@link android.view.View.DragShadowBuilder}, nor does it need to be related
     * to the View the user selected for dragging.
     * </p>
     * @param data A {@link android.content.ClipData} object pointing to the data to be
     * transferred by the drag and drop operation.
     * @param shadowBuilder A {@link android.view.View.DragShadowBuilder} object for building the
     * drag shadow.
     * @param myLocalState An {@link java.lang.Object} containing local data about the drag and
     * drop operation. When dispatching drag events to views in the same activity this object
     * will be available through {@link android.view.DragEvent#getLocalState()}. Views in other
     * activities will not have access to this data ({@link android.view.DragEvent#getLocalState()}
     * will return null).
     * <p>
     * myLocalState is a lightweight mechanism for the sending information from the dragged View
     * to the target Views. For example, it can contain flags that differentiate between a
     * a copy operation and a move operation.
     * </p>
     * @param flags Flags that control the drag and drop operation. This can be set to 0 for no
     * flags, or any combination of the following:
     *     <ul>
     *         <li>{@link #DRAG_FLAG_GLOBAL}</li>
     *         <li>{@link #DRAG_FLAG_GLOBAL_PERSISTABLE_URI_PERMISSION}</li>
     *         <li>{@link #DRAG_FLAG_GLOBAL_PREFIX_URI_PERMISSION}</li>
     *         <li>{@link #DRAG_FLAG_GLOBAL_URI_READ}</li>
     *         <li>{@link #DRAG_FLAG_GLOBAL_URI_WRITE}</li>
     *         <li>{@link #DRAG_FLAG_OPAQUE}</li>
     *     </ul>
     * @return {@code true} if the method completes successfully, or
     * {@code false} if it fails anywhere. Returning {@code false} means the system was unable to
     * do a drag, and so no drag operation is in progress.
     */
    public final boolean startDragAndDrop(ClipData data, DragShadowBuilder shadowBuilder,Object myLocalState, int flags) 
複製代碼

看到英文就頭大?沒事,我來翻譯解釋一下。git

啓動拖放操做。當應用程序調用此方法時,它將傳遞一個DragShadowBuilder對象到系統。系統調用此對象的onProvideShadowMetrics(Point, Point)方法獲取拖動陰影的參數指標,而後調用onDrawShadow(Canvas)來繪製陰影。一旦系統有了拖動陰影,它就開始拖拽操做,經過將拖拽事件發送到當前可見的應用程序中的全部視圖對象。這些視圖能夠經過設置OnDragListener在或者實現onDragEvent方法接受DragEvent(事件)來響應和拖拽事件。github

能夠看到有四個參數:api

ClipData databash

其實就是一個封裝數據的對象,經過拖放操做傳遞給接受者。該對象能夠存放一個Item的集合,Item能夠存放以下數據:app

public static class Item {
        final CharSequence mText;
        final String mHtmlText;
        final Intent mIntent;
        Uri mUri;
}
複製代碼

注意到能夠存放Intent,所以,一般能夠將參數存入intent,而後經過靜態方法直接建立ClipData對象:ide

ClipData clipData = ClipData.newIntent("label", intent);
複製代碼

該數據能夠在監聽的中的DragEvent獲取ui

ClipData clipData = event.getClipData();
複製代碼

簡單點說就是能夠將一些數據傳遞給拖拽的接受者,該拖拽其實能夠跨Activity的,若是隻是同一個Activity可使用第三個參數傳遞數據。this

DragShadowBuilder shadowBuilder

用於建立拖拽view是的陰影,也就是跟隨手指移動的視圖,一般直接使用默認便可生成與一個原始view相同,帶有透明度的陰影:

View.DragShadowBuilder shadowBuilder = new View.DragShadowBuilder(v);
複製代碼

Object myLocalState

當你的拖拽行爲是在同一個Activity中進行時能夠傳遞一個任意對象,在監聽中能夠經過{@link android.view.DragEvent#getLocalState()}得到。若是是跨Activity拖拽中沒法訪問此數據,getLocalState()將返回null。

int flags

控制拖放操做的標誌。由於沒有標誌能夠設置爲0,flag標誌拖動是否能夠跨越窗口以及一些訪問權限(須要API24+)。

瞭解了方法參數含義,接下來就是啓用拖拽了,一般會經過長按來觸發拖拽:

iv.setOnLongClickListener(new View.OnLongClickListener() {
            @Override
            public boolean onLongClick(View v) {
                View.DragShadowBuilder shadowBuilder = new View.DragShadowBuilder(v);
                v.startDrag(null, shadowBuilder, null, 0);
                //震動反饋
                v.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS, HapticFeedbackConstants.FLAG_IGNORE_GLOBAL_SETTING);
                return true;
            }
        });
複製代碼

開始拖拽後還要有來接受這些拖拽事件,這就須要OnDragListener了。

OnDragListener

OnDragListener是在View中定義的接口,用於響應拖拽事件,能夠經過View的setOnDragListener 方法設置監聽,有點相似於點擊事件。

public interface OnDragListener { 
        boolean onDrag(View v, DragEvent event);
}
複製代碼

設置監聽,實現onDrag(View v, DragEvent event)方法,其中View是設置該監聽的view,DragEvent是拖拽事件,能夠經過event.getAction() 獲取具體事件類型,這和TouchEvent很是相似,具體事件類型有以下幾種:

fl_blue.setOnDragListener(new View.OnDragListener() {
            @Override
            public boolean onDrag(View v, DragEvent event) {
                //v 永遠是設置該監聽的view,這裏即fl_blue
                String simpleName = v.getClass().getSimpleName();
                Log.w(BLUE, "view name:" + simpleName);
                
                //獲取事件
                int action = event.getAction();
                switch (action) {
                    case DragEvent.ACTION_DRAG_STARTED:
                        Log.i(BLUE, "開始拖拽");
                        break;
                    case DragEvent.ACTION_DRAG_ENDED:
                        Log.i(BLUE, "結束拖拽");
                        break;
                    case DragEvent.ACTION_DRAG_ENTERED:
                        Log.i(BLUE, "拖拽的view進入監聽的view時");
                        break;
                    case DragEvent.ACTION_DRAG_EXITED:
                        Log.i(BLUE, "拖拽的view離開監聽的view時");
                        break;
                    case DragEvent.ACTION_DRAG_LOCATION:
                        float x = event.getX();
                        float y = event.getY();
                        long l = SystemClock.currentThreadTimeMillis();
                        Log.i(BLUE, "拖拽的view在監聽view中的位置:x =" + x + ",y=" + y);
                        break;
                    case DragEvent.ACTION_DROP:
                        Log.i(BLUE, "釋放拖拽的view");
                        break;
                }
                //是否響應拖拽事件,true響應,返回false只能接受到ACTION_DRAG_STARTED事件,後續事件不會收到
                return true;
            }
        });
複製代碼

此處經過event.getX(); event.getY(); 獲取的x,y是手指(也便是被拖拽view的中心點)在監聽view的位置。

釋放手指會觸發ACTION_DRAG_ENDED 事件,若是此時被拖拽的view正好在監聽的view中,則會先觸發ACTION_DROP 事件。

這裏寫圖片描述

能夠同時有多個view設置拖拽監聽接受事件,我給紅色和藍色view都設置了OnDragListener,而後拖動Android圖片到藍色區域後釋放,能夠看到日誌以下:

03-09 14:53:54.518 12937-12937/com.huburt.app.androiddrag I/RED: 開始拖拽
03-09 14:53:54.518 12937-12937/com.huburt.app.androiddrag I/BLUE: 開始拖拽
03-09 14:53:55.689 12937-12937/com.huburt.app.androiddrag I/BLUE: 拖拽的view進入監聽的view時
03-09 14:53:55.689 12937-12937/com.huburt.app.androiddrag I/BLUE: 拖拽的view在BLUE中的位置:x =111.0,y=2.0
03-09 14:53:55.870 12937-12937/com.huburt.app.androiddrag I/BLUE: 拖拽的view在BLUE中的位置:x =112.0,y=23.0
03-09 14:53:56.014 12937-12937/com.huburt.app.androiddrag I/BLUE: 釋放拖拽的view
03-09 14:53:56.017 12937-12937/com.huburt.app.androiddrag I/RED: 結束拖拽
03-09 14:53:56.017 12937-12937/com.huburt.app.androiddrag I/BLUE: 結束拖拽
複製代碼

如今咱們已經能夠把Android圖片拖出來,可是還不能把它放入目標view,其實也挺簡單的,只須要在ACTION_DROP事件作一些處理便可:

case DragEvent.ACTION_DROP:
                        Log.i(BLUE, "釋放拖拽的view");
                        ImageView localState = (ImageView) event.getLocalState();
                        FrameLayout.LayoutParams layoutParams = new FrameLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
                        layoutParams.topMargin = (int) event.getY() - localState.getWidth() / 2;
                        layoutParams.leftMargin = (int) event.getX() - localState.getHeight() / 2;
                        ((ViewGroup) localState.getParent()).removeView(localState);
                        fl_blue.addView(localState, layoutParams);
                        break;
複製代碼

這裏由於是在同一個Activity中,我是將拖拽的view直接傳遞過來了,固然也能夠只傳遞圖片,而後在接收的view中從新new一個imageview現實圖片。

運行一下就能夠看到view能夠拖拽到目標位置了。

可能文字描述不是特別清楚,請看demo

相關文章
相關標籤/搜索