Android的换行小部件布局

发布于 2021-02-02 22:55:28

我正在尝试创建一个向用户展示一些数据的活动。数据可以分为“单词”,每个单词都是一个小部件,“单词”的序列将构成数据(“句子”?),ViewGroup小部件包含这些单词。由于“句子”中所有“单词”所需的空间将超过显示器上可用的水平空间,因此,我想像平常的文本一样包装这些“句子”。

如下代码:

public class WrapTest extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        LinearLayout l = new LinearLayout(this);
        LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(
                LinearLayout.LayoutParams.FILL_PARENT,
                LinearLayout.LayoutParams.WRAP_CONTENT);
        LinearLayout.LayoutParams mlp = new LinearLayout.LayoutParams(
                new ViewGroup.MarginLayoutParams(
                        LinearLayout.LayoutParams.WRAP_CONTENT,
                        LinearLayout.LayoutParams.WRAP_CONTENT));
        mlp.setMargins(0, 0, 2, 0);

        for (int i = 0; i < 10; i++) {
            TextView t = new TextView(this);
            t.setText("Hello");
            t.setBackgroundColor(Color.RED);
            t.setSingleLine(true);
            l.addView(t, mlp);
        }

        setContentView(l, lp);
    }
}

产生类似于左图的内容,但是我想要一种布局来呈现与右图相同的小部件。

01-have.png
01-want.png

是否有这样的布局或布局和参数的组合,还是我必须为此实现自己的ViewGroup?

关注者
0
被浏览
338
1 个回答
  • 面试哥
    面试哥 2021-02-02
    为面试而生,有面试问题,就找面试哥。

    你可以通过在build.gradle文件中添加依赖项来在项目中使用它:

    dependencies {
        compile 'com.google.android:flexbox:0.3.2'
    }
    


  • 面试哥
    面试哥 2021-02-02
    为面试而生,有面试问题,就找面试哥。

    我做了自己想要的布局,但是目前还很有限。欢迎提出意见和改进建议。

    活动:

    package se.fnord.xmms2.predicate;
    
    import se.fnord.android.layout.PredicateLayout;
    import android.app.Activity;
    import android.graphics.Color;
    import android.os.Bundle;
    import android.widget.TextView;
    
    public class Predicate extends Activity {
        /** Called when the activity is first created. */
        @Override
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
    
            PredicateLayout l = new PredicateLayout(this);
            for (int i = 0; i < 10; i++) {
                TextView t = new TextView(this);
                t.setText("Hello");
                t.setBackgroundColor(Color.RED);
                t.setSingleLine(true);
                l.addView(t, new PredicateLayout.LayoutParams(2, 0));
            }
    
            setContentView(l);
        }
    }
    

    或在XML布局中:

    <se.fnord.android.layout.PredicateLayout
        android:id="@+id/predicate_layout"
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content"
    />
    

    和布局:

    package se.fnord.android.layout;
    
    import android.content.Context;
    import android.util.AttributeSet;
    import android.view.View;
    import android.view.ViewGroup;
    
    /**
     * ViewGroup that arranges child views in a similar way to text, with them laid
     * out one line at a time and "wrapping" to the next line as needed.
     * 
     * Code licensed under CC-by-SA
     *  
     * @author Henrik Gustafsson
     * @see http://stackoverflow.com/questions/549451/line-breaking-widget-layout-for-android
     * @license http://creativecommons.org/licenses/by-sa/2.5/
     *
     */
    public class PredicateLayout extends ViewGroup {
    
        private int line_height;
    
        public PredicateLayout(Context context) {
            super(context);
        }
    
        public PredicateLayout(Context context, AttributeSet attrs){
            super(context, attrs);
        }
    
        @Override
        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
            assert(MeasureSpec.getMode(widthMeasureSpec) != MeasureSpec.UNSPECIFIED);
    
            final int width = MeasureSpec.getSize(widthMeasureSpec);
    
            // The next line is WRONG!!! Doesn't take into account requested MeasureSpec mode!
            int height = MeasureSpec.getSize(heightMeasureSpec) - getPaddingTop() - getPaddingBottom();
            final int count = getChildCount();
            int line_height = 0;
    
            int xpos = getPaddingLeft();
            int ypos = getPaddingTop();
    
            for (int i = 0; i < count; i++) {
                final View child = getChildAt(i);
                if (child.getVisibility() != GONE) {
                    final LayoutParams lp = (LayoutParams) child.getLayoutParams();
                    child.measure(
                            MeasureSpec.makeMeasureSpec(width, MeasureSpec.AT_MOST),
                            MeasureSpec.makeMeasureSpec(height, MeasureSpec.UNSPECIFIED));
    
                    final int childw = child.getMeasuredWidth();
                    line_height = Math.max(line_height, child.getMeasuredHeight() + lp.height);
    
                    if (xpos + childw > width) {
                        xpos = getPaddingLeft();
                        ypos += line_height;
                    }
    
                    xpos += childw + lp.width;
                }
            }
            this.line_height = line_height;
    
            if (MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.UNSPECIFIED){
                height = ypos + line_height;
    
            } else if (MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.AT_MOST){
                if (ypos + line_height < height){
                    height = ypos + line_height;
                }
            }
            setMeasuredDimension(width, height);
        }
    
        @Override
        protected LayoutParams generateDefaultLayoutParams() {
            return new LayoutParams(1, 1); // default of 1px spacing
        }
    
        @Override
        protected boolean checkLayoutParams(LayoutParams p) {
            return (p instanceof LayoutParams);
        }
    
        @Override
        protected void onLayout(boolean changed, int l, int t, int r, int b) {
            final int count = getChildCount();
            final int width = r - l;
            int xpos = getPaddingLeft();
            int ypos = getPaddingTop();
    
            for (int i = 0; i < count; i++) {
                final View child = getChildAt(i);
                if (child.getVisibility() != GONE) {
                    final int childw = child.getMeasuredWidth();
                    final int childh = child.getMeasuredHeight();
                    final LayoutParams lp = (LayoutParams) child.getLayoutParams();
                    if (xpos + childw > width) {
                        xpos = getPaddingLeft();
                        ypos += line_height;
                    }
                    child.layout(xpos, ypos, xpos + childw, ypos + childh);
                    xpos += childw + lp.width;
                }
            }
        }
    }
    

    结果:

    01-have2.png



知识点
面圈网VIP题库

面圈网VIP题库全新上线,海量真题题库资源。 90大类考试,超10万份考试真题开放下载啦

去下载看看