/* * Copyright 2013 Chris Banes * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package uk.co.senab.actionbarpulltorefresh.library; import android.content.Context; import android.content.res.Configuration; import android.util.AttributeSet; import android.util.Log; import android.view.MotionEvent; import android.view.View; import android.widget.FrameLayout; /** * FIXME */ public class PullToRefreshLayout extends FrameLayout { private static final boolean DEBUG = false; private static final String LOG_TAG = "PullToRefreshLayout"; private PullToRefreshAttacher mPullToRefreshAttacher; private View mCurrentTouchTarget; public PullToRefreshLayout(Context context) { this(context, null); } public PullToRefreshLayout(Context context, AttributeSet attrs) { this(context, attrs, 0); } public PullToRefreshLayout(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } /** * Set the {@link PullToRefreshAttacher} to be used with this layout. The view which is added * to this layout will automatically be added as a refreshable-view in the attacher. */ public void setPullToRefreshAttacher(PullToRefreshAttacher attacher, PullToRefreshAttacher.OnRefreshListener refreshListener) { View view; for (int i = 0, z = getChildCount(); i < z; i++) { view = getChildAt(i); if (mPullToRefreshAttacher != null) { mPullToRefreshAttacher.removeRefreshableView(view); } if (attacher != null) { if (DEBUG) Log.d(LOG_TAG, "Adding View to Attacher: " + view); attacher.addRefreshableView(view, null, refreshListener, false); } } mPullToRefreshAttacher = attacher; } @Override public boolean onInterceptTouchEvent(MotionEvent event) { if (DEBUG) Log.d(LOG_TAG, "onInterceptTouchEvent. " + event.toString()); if (mPullToRefreshAttacher != null && getChildCount() > 0) { View target = getChildForTouchEvent(event); if (target != null && mPullToRefreshAttacher.onInterceptTouchEvent(target, event)) { mCurrentTouchTarget = target; return true; } } // Reset Current Touch Target mCurrentTouchTarget = null; return false; } @Override public boolean onTouchEvent(MotionEvent event) { if (DEBUG) Log.d(LOG_TAG, "onTouchEvent. " + event.toString()); if (mPullToRefreshAttacher != null) { // This is an edge-case. If the ViewGroup does not contain a valid touch target then // Android calls onTouchEvent after onInterceptTouchEvent with ACTION_DOWN event. // If that happens then we need to find the visible view and pass it to the attacher as // usual. if (mCurrentTouchTarget == null && event.getAction() == MotionEvent.ACTION_DOWN) { mCurrentTouchTarget = getChildForTouchEvent(event); } if (mCurrentTouchTarget != null) { return mPullToRefreshAttacher.onTouchEvent(mCurrentTouchTarget, event); } } // Reset Current Touch Target mCurrentTouchTarget = null; return super.onTouchEvent(event); } @Override protected void onConfigurationChanged(Configuration newConfig) { super.onConfigurationChanged(newConfig); if (mPullToRefreshAttacher != null) { mPullToRefreshAttacher.onConfigurationChanged(newConfig); } } private View getChildForTouchEvent(MotionEvent event) { final float x = event.getX(), y = event.getY(); View child; for (int z = getChildCount() - 1; z >= 0 ; z--) { child = getChildAt(z); if (child.isShown() && x >= child.getLeft() && x <= child.getRight() && y >= child.getTop() && y <= child.getBottom()) { if (DEBUG) Log.d(LOG_TAG, "Got Child for Touch Event: " + child); return child; } } return null; } }