首页 热点专区 小学知识 中学知识 出国留学 考研考公
您的当前位置:首页正文

写一个Anroid通用style的详情页基类

2024-12-09 来源:要发发知识网

开发中经常需要写一个带tab切换并且tab有吸顶效果而且头要渐变效果的详情页,如下图:
![


TT8(K5X~CK$NH(Z@VWC%0PU.png

之前做项目时,一个人需要维护三四种这样类似的页面,每次在scrollView滚动时做一些列的计算都很痛苦,各种计算吸顶以及滚动时tab选中的位置。后来就萌生了一个想法,直接写一个基类,在基类里面完成完成这一系列的计算,然后子类直接继承基类,实现几个画页面的方法就好了!下面直接上代码:

基类activity:

private View mActionBarView;
    private LinearLayout mContainerLayout;
    private ObservableScrollView mScrollView;
    private CommonDetailTabLayout mTabLayout1;
    private CommonDetailTabLayout mTabLayout2;
    protected ViewGroup mBottomContainer;

    private int mStatusBarHeight = 0;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.lesson_detail_layout);
        initView();
    }

    private void initView() {
        int resourceId = getResources().getIdentifier("status_bar_height", "dimen", "android");
        mStatusBarHeight = getResources().getDimensionPixelSize(resourceId);
        initTabLayout();
        mActionBarView = getView(R.id.ll_actionbar);
        mActionBarView.getBackground().setAlpha(0);
        mScrollView = getView(R.id.sv_content);
        mScrollView.addScrollViewCallbacks(mCallBacks);
        mContainerLayout = getView(R.id.ll_container);
        if (getHeaderView() != null) {
            mContainerLayout.addView(getHeaderView());
        }
        mContainerLayout.addView(mTabLayout2);
        for (View tabView : getTabViews()) {
            if (tabView != null) {
                mContainerLayout.addView(tabView);
            }
        }
        mBottomContainer = getView(R.id.ll_bottom_container);
        if (getBottomView() == null) {
            mBottomContainer.setVisibility(View.GONE);
        } else {
            mBottomContainer.setVisibility(View.VISIBLE);
        }

    }

    protected abstract View getHeaderImageView();

    protected abstract View getHeaderView();

    protected abstract View getBottomView();

    protected abstract ArrayList<View> getTabViews();

    protected void setTabData(String... data) {
        ArrayList<CommonDetailTabLayout.TabObject> tabData = new ArrayList<>();
        int length = data.length;
        if (length != getTabViews().size()) {
            return;

        }
        for (int i = 0; i < length; i++) {
            CommonDetailTabLayout.TabObject tabObject = new CommonDetailTabLayout.TabObject();
            tabObject.tabName = data[i];
            final int finalI = i;
            tabObject.listener = new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    int height = getHeaderView().getMeasuredHeight()  - DimenUtils.dip2px(mActivity,52);
                    for (int j = 0; j < finalI ; j ++){
                        height += getTabViews().get(j).getMeasuredHeight();
                    }
                    mTabLayout1.check(finalI);
                    mTabLayout2.check(finalI);
                    mScrollView.scrollVerticallyTo(height);
                }
            };
            tabData.add(tabObject);
        }
        mTabLayout1.setData(tabData);
        mTabLayout2.setData(tabData);
    }

    private void initTabLayout() {
        mTabLayout1 = getView(R.id.tab);
        mTabLayout1.setVisibility(View.GONE);
        mTabLayout2 = new CommonDetailTabLayout(mActivity);
    }

    private ObservableScrollViewCallbacks mCallBacks = new ObservableScrollViewCallbacks() {
        @Override
        public void onScrollChanged(int scrollY, boolean firstScroll, boolean dragging) {
            //计算actionbar透明度
            float progress = (scrollY * 1.0f)
                    / (getHeaderImageView().getHeight() - DimenUtils.dip2px(mActivity,52));
            progress = progress > 1 ? 1 : progress;
            mActionBarView.getBackground().setAlpha((int) (255 * progress));

            //滚动选中tab
            int size = getTabViews().size();
            for (int i = 0; i < size; i++) {
                View view = getTabViews().get(i);
                int top = view.getTop() - DimenUtils.dip2px(mActivity, 94);
                int bottom = top + view.getMeasuredHeight();
                if (scrollY > top && scrollY < bottom) {
                    mTabLayout1.check(i);
                    mTabLayout2.check(i);
                    break;
                }
                if (i == size - 2 && scrollY > top) {
                    mTabLayout1.check(size - 1);
                    mTabLayout2.check(size - 1);
                }
            }
            //控制顶部tab的显示与隐藏
            int headerHeight = DimenUtils.dip2px(mActivity, 52) + mStatusBarHeight;
            int[] loc = new int[2];
            mTabLayout2.getLocationOnScreen(loc);
            mTabLayout1.setVisibility(loc[1] >= headerHeight ? View.GONE : View.VISIBLE);

        }

        @Override
        public void onDownMotionEvent() {

        }

        @Override
        public void onUpOrCancelMotionEvent(ScrollState scrollState) {

        }
    };

tab控件

public class CommonDetailTabLayout extends RadioGroup {
    private Context mContext;
    public ArrayList<RadioButton> mList = new ArrayList<RadioButton>();

    public CommonDetailTabLayout(Context context) {
        super(context);
        mContext = context;
        initConfig();
    }

    public CommonDetailTabLayout(Context context, AttributeSet attributeSet) {
        super(context, attributeSet);
        mContext = context;
        initConfig();
    }

    private void initConfig() {
        setOrientation(HORIZONTAL);
        setLayoutParams(new AbsListView.LayoutParams(AbsListView.LayoutParams.MATCH_PARENT, Utils.dip2px(mContext, 42)));
        int padding = Utils.dip2px(mContext,10);
        setBackgroundResource(R.drawable.bg_common_twoline);
        setPadding(padding, 0, padding, 0);
    }

    public void setData(ArrayList<TabObject> list) {
        if (list == null) {
            return;
        }
        removeAllViews();
        mList.clear();
        int size = list.size();
        for (int i = 0; i < size; i++) {
            RadioButton radio = (RadioButton) LayoutInflater.from(mContext).inflate(R.layout.detai_radio_button, this, false);
            radio.setId(i);
            if (i == 0) {
                radio.setChecked(true);
            }
            radio.setText(list.get(i).tabName);
            radio.setOnClickListener(list.get(i).listener);
            mList.add(radio);
            addView(radio);
        }
    }

    public static class TabObject {
        public String tabName;
        public OnClickListener listener;
    }

    public ArrayList<RadioButton> getRadioList() {
        return mList;
    }

    public void resetChecked() {
        check(getChildAt(0).getId());
    }

}

重写的scrollview

public class ObservableScrollView extends ScrollView implements ObserveScrollable {

    // Fields that should be saved onSaveInstanceState
    private int mPrevScrollY;
    private int mScrollY;

    // Fields that don't need to be saved onSaveInstanceState
    private ObservableScrollViewCallbacks mCallbacks;
    private List<ObservableScrollViewCallbacks> mCallbackCollection;
    private ScrollState mScrollState;
    private boolean mFirstScroll;
    private boolean mDragging;
    private boolean mIntercepted;
    private MotionEvent mPrevMoveEvent;
    private ViewGroup mTouchInterceptionViewGroup;

    public ObservableScrollView(Context context) {
        super(context);
    }

    public ObservableScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public ObservableScrollView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    @Override
    public void onRestoreInstanceState(Parcelable state) {
        SavedState ss = (SavedState) state;
        mPrevScrollY = ss.prevScrollY;
        mScrollY = ss.scrollY;
        super.onRestoreInstanceState(ss.getSuperState());
    }

    @Override
    public Parcelable onSaveInstanceState() {
        Parcelable superState = super.onSaveInstanceState();
        SavedState ss = new SavedState(superState);
        ss.prevScrollY = mPrevScrollY;
        ss.scrollY = mScrollY;
        return ss;
    }

    @Override
    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        super.onScrollChanged(l, t, oldl, oldt);
        if (hasNoCallbacks()) {
            return;
        }
        mScrollY = t;

        dispatchOnScrollChanged(t, mFirstScroll, mDragging);
        if (mFirstScroll) {
            mFirstScroll = false;
        }

        if (mPrevScrollY < t) {
            mScrollState = ScrollState.UP;
        } else if (t < mPrevScrollY) {
            mScrollState = ScrollState.DOWN;
            //} else {
            // Keep previous state while dragging.
            // Never makes it STOP even if scrollY not changed.
            // Before Android 4.4, onTouchEvent calls onScrollChanged directly for ACTION_MOVE,
            // which makes mScrollState always STOP when onUpOrCancelMotionEvent is called.
            // STOP state is now meaningless for ScrollView.
        }
        mPrevScrollY = t;
    }

    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        if (hasNoCallbacks()) {
            return super.onInterceptTouchEvent(ev);
        }
        switch (ev.getActionMasked()) {
            case MotionEvent.ACTION_DOWN:
                // Whether or not motion events are consumed by children,
                // flag initializations which are related to ACTION_DOWN events should be executed.
                // Because if the ACTION_DOWN is consumed by children and only ACTION_MOVEs are
                // passed to parent (this view), the flags will be invalid.
                // Also, applications might implement initialization codes to onDownMotionEvent,
                // so call it here.
                mFirstScroll = mDragging = true;
                dispatchOnDownMotionEvent();
                break;
        }
        return super.onInterceptTouchEvent(ev);
    }

    @Override
    public boolean onTouchEvent(MotionEvent ev) {
        if (hasNoCallbacks()) {
            return super.onTouchEvent(ev);
        }
        switch (ev.getActionMasked()) {
            case MotionEvent.ACTION_UP:
            case MotionEvent.ACTION_CANCEL:
                mIntercepted = false;
                mDragging = false;
                dispatchOnUpOrCancelMotionEvent(mScrollState);
                break;
            case MotionEvent.ACTION_MOVE:
                if (mPrevMoveEvent == null) {
                    mPrevMoveEvent = ev;
                }
                float diffY = ev.getY() - mPrevMoveEvent.getY();
                mPrevMoveEvent = MotionEvent.obtainNoHistory(ev);
                if (getCurrentScrollY() - diffY <= 0) {
                    // Can't scroll anymore.

                    if (mIntercepted) {
                        // Already dispatched ACTION_DOWN event to parents, so stop here.
                        return false;
                    }

                    // Apps can set the interception target other than the direct parent.
                    final ViewGroup parent;
                    if (mTouchInterceptionViewGroup == null) {
                        parent = (ViewGroup) getParent();
                    } else {
                        parent = mTouchInterceptionViewGroup;
                    }

                    // Get offset to parents. If the parent is not the direct parent,
                    // we should aggregate offsets from all of the parents.
                    float offsetX = 0;
                    float offsetY = 0;
                    for (View v = this; v != null && v != parent; v = (View) v.getParent()) {
                        offsetX += v.getLeft() - v.getScrollX();
                        offsetY += v.getTop() - v.getScrollY();
                    }
                    final MotionEvent event = MotionEvent.obtainNoHistory(ev);
                    event.offsetLocation(offsetX, offsetY);

                    if (parent.onInterceptTouchEvent(event)) {
                        mIntercepted = true;

                        // If the parent wants to intercept ACTION_MOVE events,
                        // we pass ACTION_DOWN event to the parent
                        // as if these touch events just have began now.
                        event.setAction(MotionEvent.ACTION_DOWN);

                        // Return this onTouchEvent() first and set ACTION_DOWN event for parent
                        // to the queue, to keep events sequence.
                        post(new Runnable() {
                            @Override
                            public void run() {
                                parent.dispatchTouchEvent(event);
                            }
                        });
                        return false;
                    }
                    // Even when this can't be scrolled anymore,
                    // simply returning false here may cause subView's click,
                    // so delegate it to super.
                    return super.onTouchEvent(ev);
                }
                break;
        }
        return super.onTouchEvent(ev);
    }

    @Override
    public void setScrollViewCallbacks(ObservableScrollViewCallbacks listener) {
        mCallbacks = listener;
    }

    @Override
    public void addScrollViewCallbacks(ObservableScrollViewCallbacks listener) {
        if (mCallbackCollection == null) {
            mCallbackCollection = new ArrayList<ObservableScrollViewCallbacks>();
        }
        mCallbackCollection.add(listener);
    }

    @Override
    public void removeScrollViewCallbacks(ObservableScrollViewCallbacks listener) {
        if (mCallbackCollection != null) {
            mCallbackCollection.remove(listener);
        }
    }

    @Override
    public void clearScrollViewCallbacks() {
        if (mCallbackCollection != null) {
            mCallbackCollection.clear();
        }
    }

    @Override
    public void setTouchInterceptionViewGroup(ViewGroup viewGroup) {
        mTouchInterceptionViewGroup = viewGroup;
    }

    @Override
    public void scrollVerticallyTo(int y) {
        scrollTo(0, y);
    }

    @Override
    public int getCurrentScrollY() {
        return mScrollY;
    }

    private void dispatchOnDownMotionEvent() {
        if (mCallbacks != null) {
            mCallbacks.onDownMotionEvent();
        }
        if (mCallbackCollection != null) {
            for (int i = 0; i < mCallbackCollection.size(); i++) {
                ObservableScrollViewCallbacks callbacks = mCallbackCollection.get(i);
                callbacks.onDownMotionEvent();
            }
        }
    }

    private void dispatchOnScrollChanged(int scrollY, boolean firstScroll, boolean dragging) {
        if (mCallbacks != null) {
            mCallbacks.onScrollChanged(scrollY, firstScroll, dragging);
        }
        if (mCallbackCollection != null) {
            for (int i = 0; i < mCallbackCollection.size(); i++) {
                ObservableScrollViewCallbacks callbacks = mCallbackCollection.get(i);
                callbacks.onScrollChanged(scrollY, firstScroll, dragging);
            }
        }
    }
    private void dispatchOnUpOrCancelMotionEvent(ScrollState scrollState) {
        if (mCallbacks != null) {
            mCallbacks.onUpOrCancelMotionEvent(scrollState);
        }
        if (mCallbackCollection != null) {
            for (int i = 0; i < mCallbackCollection.size(); i++) {
                ObservableScrollViewCallbacks callbacks = mCallbackCollection.get(i);
                callbacks.onUpOrCancelMotionEvent(scrollState);
            }
        }
    }

    private boolean hasNoCallbacks() {
        return mCallbacks == null && mCallbackCollection == null;
    }

    static class SavedState extends BaseSavedState {
        int prevScrollY;
        int scrollY;

        /**
         * Called by onSaveInstanceState.
         */
        SavedState(Parcelable superState) {
            super(superState);
        }

        /**
         * Called by CREATOR.
         */
        private SavedState(Parcel in) {
            super(in);
            prevScrollY = in.readInt();
            scrollY = in.readInt();
        }

        @Override
        public void writeToParcel(Parcel out, int flags) {
            super.writeToParcel(out, flags);
            out.writeInt(prevScrollY);
            out.writeInt(scrollY);
        }

        public static final Creator<SavedState> CREATOR
                = new Creator<SavedState>() {
            @Override
            public SavedState createFromParcel(Parcel in) {
                return new SavedState(in);
            }

            @Override
            public SavedState[] newArray(int size) {
                return new SavedState[size];
            }
        };
    }

}

基类activity的layout

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout 
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <ObservableScrollView
        android:id="@+id/sv_content"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:scrollbars="none">
        <LinearLayout
            android:id="@+id/ll_container"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical">
        </LinearLayout>
    </ObservableScrollView>

    <include
        android:id="@+id/ll_actionbar"
        layout="@layout/edu_actionbar_layout" />

    <CommonDetailTabLayout
        android:id="@+id/tab"
        android:layout_width="match_parent"
        android:layout_height="42dp"
        android:layout_below="@id/ll_actionbar">

    <CommonDetailTabLayout>
</RelativeLayout>

下面是我自己继承基类写的一个小demo

public class DepositDetailActivity extends DetailBaseActivity {
    private DetailHeaderWidget mHeaderWidget;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        mHeaderWidget = new DetailHeaderWidget(this);
        super.onCreate(savedInstanceState);
        setTitle("课程详情");
        setTabData("1111","2222","3333");
    }

    @Override
    protected View getHeaderImageView() {
        return mHeaderWidget.getTopImage();
    }

    @Override
    protected View getHeaderView() {
        return mHeaderWidget.getView();
    }

    @Override
    protected ArrayList<View> getTabViews() {
        ArrayList<View> list = new ArrayList<>();
        for (int i = 0 ; i < 3 ; i++){
            TextView textView = new TextView(mActivity);
            textView.setText("第" + i);
            textView.setGravity(Gravity.CENTER);
            textView.setLayoutParams(new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, Utils.dip2px(mActivity,i == 2 ? 800 : 800)));
            list.add(textView);
        }
        return list;
    }
}

效果如图:

显示全文