可水平居中的TextView

如上图所示:多个TextView一行内显示不下,可水平拖动,点击某个TextView后,可以屏幕居中,则居中。

curTv.getLeft()为相对于起始点的绝对坐标,值可以大于屏幕的宽度。

HorizontalScrollView内LineLayout的宽度,在fragment的onViewCreated方法中无法获取,要调用mLineLayout.post方法才可。

java:


public class FragmentProject extends BaseFragment {
    private static final String TAG = "FragmentProject";
    private Context mContext;
    private View mRootView;
    private FragmentActivity mActivity;

    private ProjectTitleScrollView mScrollView;
    private LinearLayout mLineLayout;
    private ViewPager mVp;
    //FragmentStatePagerAdapter会回收fragment,会执行onDestroyView、onDestroy方法,可很好的控制内存
    private FragmentStatePagerAdapter mAdapter;
    private SparseArray<FragmentProjectArticleList> mFrList = new SparseArray<>();

    //scrCx : 屏幕的中轴线
    //scrollOverX : 滑动后,偏移的x坐标
    private int scrCx, scrollOverX;
    
    @Nullable
    @Override
    public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        mRootView = inflater.inflate(R.layout.fragment_project, container, false);
        return mRootView;
    }

    @Override
    public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);
        mContext = getContext();
        mActivity = getActivity();

        initView();
        initDate();
    }

    private void initView() {
        mLineLayout = mRootView.findViewById(R.id.ll_text);
        mScrollView = mRootView.findViewById(R.id.scrollView);
        mVp = mRootView.findViewById(R.id.project_vp);

        mScrollView.setListener(new ScrollViewListener() {
            @Override
            public void onScrollChanged(HorizontalScrollView scrollView, int l, int t, int oldl, int oldt) {
//                Log.d(TAG, "onScrollChanged: oldl =" + oldl + ", l = " + l);
                scrollOverX = l;
            }
        });
    }

    private ProjectPresenter mPresenter;
    private ArrayList<ProjectResult.DataBean> mDataList = new ArrayList<>();

    private void initDate() {
		showHScrollTextView();
    }

    private void showHScrollTextView(){
        scrCx = Utils.getScreenW(mContext)/2;
//        Log.d(TAG, "initDate: scrCx = " + scrCx);
//
         ArrayList<String> list = new ArrayList<>();

        list.add("项目1");
        list.add("项目2");
        list.add("项目3");
        list.add("项目1333");
        list.add("项目14444");
        list.add("项目15556");
        list.add("项目15555555577777");
        list.add("项目1555555558888");
        list.add("项目");
        list.add("项目1");

        LinearLayout.LayoutParams layoutParams = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
        layoutParams.setMargins(Utils.dp2px(mContext,15),0,Utils.dp2px(mContext,15),0);//4个参数按顺序分别是左上右下

        for (int i = 0; i < list.size(); i++){
            TextView tv = new TextView(mContext);
            tv.setText(list.get(i));
            tv.setTextSize(TypedValue.COMPLEX_UNIT_SP, 20);

            tv.setTextColor(i == 0 ? Color.WHITE : getResources().getColor(R.color.little_dark));
            tv.setLayoutParams(layoutParams);
            tv.setTag(i);
            mLineLayout.addView(tv);

            tv.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View view) {
                    onItemClick(view);
                }
            });
        }

        //只写mLineLayout.getRight(),获取的值为0,此时mLineLayout还没绘制完成,得不到值
        //https://www.cnblogs.com/ouyangping/p/7398753.html
//        mLineLayout.post(new Runnable() {
//            @Override
//            public void run() {
//                LINEAR_LAYOUT_W = mLineLayout.getRight();
                Log.d(TAG, "initDate: LINEAR_LAYOUT_W = " + LINEAR_LAYOUT_W + ", " + mLineLayout.getWidth());
//            }
//        });
    }

    private void onItemClick(View view){
        int index = (int)((TextView)view).getTag();
        TextView curTv = (TextView)mLineLayout.getChildAt(index);
        //getLeft 和onLayout之后的getWidth的关系?都是在布局之后,可以得到其值的吗?
        Log.d(TAG, "onClick: index = " + index + ", " + curTv.getText()
                + ", 范围[" + curTv.getLeft() + "," + curTv.getRight() + "]"
//                + ", all_l = " + mScrollView.getLeft() + ", all_r = " + mScrollView.getRight()
//                + ", l_l = " + mLineLayout.getLeft() + ", l_r = " + mLineLayout.getRight()
//                + ", l_w = " + mLineLayout.getWidth() + ", margin_l/r = " + Utils.dp2px(mContext,30)
        );

        //textAbsCx : text的中轴线到scrollView的绝对距离
        int textAbsCx = (curTv.getRight()+curTv.getLeft())/2;
        Log.d(TAG, "onItemClick: textAbsCx = " + textAbsCx + ", scrollOverX = " + scrollOverX);

        //当前text的中心轴线距离屏幕左边界的距离,也就是text中轴线的值
        int textCx2ScrLDis = textAbsCx - scrollOverX;
        Log.d(TAG, "textCx2ScrLDis = " + textCx2ScrLDis + ", scrCx = " + scrCx);
//        if (textCx2ScrLDis > scrCx){//text中轴线在屏幕中轴线右侧
//            //text是否可以向左移动到屏幕中轴线处:都可以
//            mScrollView.smoothScrollBy(textCx2ScrLDis-scrCx, 0);
//        } else {//左侧
//            //text是否可以向右移动到屏幕中轴线处
//            if (scrollOverX < scrCx) {
//                if (textAbsCx > scrCx) {//可以
//                    mScrollView.smoothScrollBy(textCx2ScrLDis-scrCx, 0);
//                } else {//回到最开始
//                    mScrollView.smoothScrollTo(0, 0);
//                }
//            } else {//可以
//                mScrollView.smoothScrollBy(textCx2ScrLDis-scrCx, 0);
//            }
//        }

        if (textCx2ScrLDis < scrCx && scrollOverX < scrCx && textAbsCx < scrCx) {
            mScrollView.smoothScrollTo(0, 0);
        } else {
            mScrollView.smoothScrollBy(textCx2ScrLDis-scrCx, 0);
        }

        for (int i = 0; i < mLineLayout.getChildCount(); i++){
            TextView temTv = (TextView) mLineLayout.getChildAt(i);
            temTv.setTextColor(i == index ? Color.WHITE : getResources().getColor(R.color.little_dark));
        }
    }
}

xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.lyl.wanandroid.view.ProjectTitleScrollView
        android:layout_width="wrap_content"
        android:layout_height="@dimen/titleBarH"
        android:scrollbars="none"
        android:background="@color/colorPrimary"
        android:id="@+id/scrollView">
        <LinearLayout
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:id="@+id/ll_text"
            android:orientation="horizontal"
            android:layout_gravity="center_vertical"
            android:paddingLeft="15dp"
            android:paddingRight="15dp"/>
<!--        android:layout_gravity="center_vertical"
layout_gravity是设置自身相当于父容器的对齐方式
-->
<!--        android:gravity="center_vertical"-->
    </com.lyl.wanandroid.view.ProjectTitleScrollView>

    <android.support.v4.view.ViewPager
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:id="@+id/project_vp"/>
</LinearLayout>
public interface ScrollViewListener {
    void onScrollChanged(HorizontalScrollView scrollView, int l, int t, int oldl, int oldt);
}

 

ps : ScrollView和HorizontalScrollView

ScrollView就是一个可以滚动的View,这个滚动的方向是垂直方向的,而HorizontalScrollView则是一个水平方向的可以滚动的View。二者只是两种滚动方向不同的View而已,其他方面都基本相同。

屏幕大小总是有限制的,对移动设备来说更是如此。当有很多内容需要显示的时候,一屏显示不完时,就需要使用滚动的方式。

ScrollView只能包含一个直接子view,这是因为ScrollView是FrameLayout的派生类,通常情况下,这个直接子view是一个LinearLayout,在直接子view(比如LinearLayout)中,可以再包含其他对象。

对于ScrollView来说,因为其是垂直方向上的滚动布局,因此通常我们给其添加一个LinearLayout的子元素,并且设置orientation为vertical(垂直方向的)

如果要屏幕支持垂直滚动和水平滚动,那么就要让HorizontalScrollView作为ScrollView的直接子view,或者让ScrollView作为HorizontalScrollView的直接子view。


Android UI系列-----ScrollView和HorizontalScrollView
ScrollView/HorizontalScrollView常用技巧
 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值