android流式布局的写法

package com.carryme.app.widget;

import android.content.Context;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;

import java.util.HashMap;
import java.util.Map;

/**
 * 流式布局
 */
public class FlowLayout extends ViewGroup {

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

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

    public FlowLayout(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        //   super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //遍历去调用所有子元素的measure方法(child.getMeasuredHeight()才能获取到值,否则为0)
        measureChildren(widthMeasureSpec, heightMeasureSpec);

        int measuredWidth = 0, measuredHeight = 0;

        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int widtMode = MeasureSpec.getMode(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        //由于计算子view所占宽度,这里传值需要自身减去PaddingRight宽度,PaddingLeft会在接下来计算子元素位置时加上
        Map<String, Integer> compute = compute(widthSize-getPaddingRight());

        //EXACTLY模式:对应于给定大小或者match_parent情况
        if (widtMode == MeasureSpec.EXACTLY) {
            measuredWidth = widthSize;
            //AT_MOS模式:对应wrap-content(需要手动计算大小,否则相当于match_parent)
        } else if (widtMode == MeasureSpec.AT_MOST) {
            measuredWidth = compute.get("allChildWidth");
        }

        if (heightMode == MeasureSpec.EXACTLY) {
            measuredHeight = heightSize;
        } else if (heightMode == MeasureSpec.AT_MOST) {
            measuredHeight = compute.get("allChildHeight");
        }
        //设置flow的宽高
        setMeasuredDimension(measuredWidth,measuredHeight);
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        for (int i = 0; i < getChildCount(); i++) {
            View child = getChildAt(i);
            Rect rect = (Rect) getChildAt(i).getTag();
            child.layout(rect.left,rect.top,rect.right,rect.bottom);
        }

    }

    /**
     * 测量过程
     * @param flowWidth 该view的宽度
     * @return  返回子元素总所占宽度和高度(用于计算Flowlayout的AT_MOST模式设置宽高)
     */
    private Map<String, Integer> compute(int flowWidth) {
        //是否是单行
        boolean aRow = true;
        MarginLayoutParams marginParams;//子元素margin
        int rowsWidth = getPaddingLeft();//当前行已占宽度(注意需要加上paddingLeft)
        int columnHeight =getPaddingTop();//当前行顶部已占高度(注意需要加上paddingTop)
        int rowsMaxHeight = 0;//当前行所有子元素的最大高度(用于换行累加高度)

        for (int i = 0; i <  getChildCount(); i++) {

            View child = getChildAt(i);
            //获取元素测量宽度和高度
            int measuredWidth = child.getMeasuredWidth();
            int measuredHeight = child.getMeasuredHeight();
            //获取元素的margin
            marginParams = (MarginLayoutParams) child.getLayoutParams();
            //子元素所占宽度 = MarginLeft+ child.getMeasuredWidth+MarginRight  注意此时不能child.getWidth,因为界面没有绘制完成,此时wdith为0
            int childWidth = marginParams.leftMargin + marginParams.rightMargin + measuredWidth;
            int childHeight = marginParams.topMargin + marginParams.bottomMargin + measuredHeight;
            //判断是否换行: 该行已占大小+该元素大小>父容器宽度  则换行

            rowsMaxHeight = Math.max(rowsMaxHeight, childHeight);
            //换行
            if (rowsWidth + childWidth > flowWidth) {
                //重置行宽度
                rowsWidth = getPaddingLeft()+getPaddingRight();
                //累加上该行子元素最大高度
                columnHeight += rowsMaxHeight;
                //重置该行最大高度
                rowsMaxHeight = childHeight;
                aRow = false;
            }
            //累加上该行子元素宽度
            rowsWidth += childWidth;
            //判断时占的宽段时加上margin计算,设置顶点位置时不包括margin位置,不然margin会不起作用,这是给View设置tag,在onlayout给子元素设置位置再遍历取出
            child.setTag(new Rect(rowsWidth - childWidth + marginParams.leftMargin, columnHeight + marginParams.topMargin, rowsWidth - marginParams.rightMargin, columnHeight + childHeight - marginParams.bottomMargin));
        }

        //返回子元素总所占宽度和高度(用于计算Flowlayout的AT_MOST模式设置宽高)
        Map<String, Integer> flowMap = new HashMap<>();
        //单行
        if (aRow) {
            flowMap.put("allChildWidth", rowsWidth);
        } else {
            //多行
            flowMap.put("allChildWidth", flowWidth);
        }
        //FlowLayout测量高度 = 当前行顶部已占高度 +当前行内子元素最大高度+FlowLayout的PaddingBottom
        flowMap.put("allChildHeight", columnHeight+rowsMaxHeight+getPaddingBottom());
        return  flowMap;
    }
}

用法

 private void addFlowLayout(AdapterSellOnSaleBinding binding, List<String> sourceList, int position) {
        LinearLayout.LayoutParams itemLayoutParams = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
        itemLayoutParams.setMargins(10, 5, 10, 5);

        FlowLayout flowLayout = new FlowLayout(mContext);

        TextView tv = new TextView(mContext);
        tv.setText("···");
        tv.setPadding(24, 5, 24, 5);
        tv.setBackground(mContext.getDrawable(R.mipmap.bg_appraisal_brand));
        tv.setTextColor(mContext.getResources().getColor(R.color.white));
        tv.setTextSize(11);

        // 按2行算,先得到FlowLayout高度 高度 = (top + bottom + viewHeight) * 2
        int tvHeight = unDisplayViewSize(tv)[1];
        int tvWidth = unDisplayViewSize(tv)[0];
        double s =  (double)sourceList.size()/4;
        int line = (int)Math.ceil(s);
        int flowHeight = (5 + 5 + tvHeight + 10) * line;

        RelativeLayout.LayoutParams layoutParams = new RelativeLayout.LayoutParams(RelativeLayout.LayoutParams.MATCH_PARENT, flowHeight);
        layoutParams.addRule(RelativeLayout.BELOW, binding.tvTime.getId());
        layoutParams.setMargins(ScreenUtil.dip2px(mContext, 8), ScreenUtil.dip2px(mContext, 4), ScreenUtil.dip2px(mContext, 8), ScreenUtil.dip2px(mContext, 4));
        // 把flowLayout添加进item
//        flowLayout.setBackgroundColor(mContext.getResources().getColor(R.color.red_ff726a));
        binding.rlSize.addView(flowLayout, layoutParams);

        flowLayout.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
            @Override
            public void onGlobalLayout() {
                flowLayout.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                List<String> interceptList = interceptList(sourceList, flowLayout.getWidth(), tvWidth);
                for(String str : interceptList) {
                    TextView tv = new TextView(mContext);
                    tv.setText(str);
                    tv.setPadding(24, 6, 24, 6);
                    tv.setBackground(mContext.getDrawable(R.mipmap.subtract));
                    tv.setTextColor(mContext.getResources().getColor(R.color.color_7F7F8C));
                    tv.setTextSize(11);
                    flowLayout.addView(tv, itemLayoutParams);
                }
            }
        });

    }

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android 中使用 Socket 进行通讯可以分为服务端和客户端两种角色,下面分别介绍它们的写法: ### 服务端 服务端需要监听一个端口,并在有客户端连接时接受客户端的请求并返回响应。以下是服务端的基本写法: ```java try { // 创建一个 ServerSocket 对象,指定监听的端口号 ServerSocket serverSocket = new ServerSocket(8888); Log.d("Server", "Server started: " + serverSocket); while (true) { // 接受客户端的连接请求,返回一个 Socket 对象 Socket clientSocket = serverSocket.accept(); Log.d("Server", "Client connected: " + clientSocket); // 从输入中读取客户端发送的数据 InputStream inputStream = clientSocket.getInputStream(); BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream)); String request = reader.readLine(); // 读取一行数据 // 向输出中写入响应数据 OutputStream outputStream = clientSocket.getOutputStream(); outputStream.write("Hello, client!\n".getBytes()); // 关闭连接 clientSocket.close(); } } catch (IOException e) { e.printStackTrace(); } ``` ### 客户端 客户端需要连接到服务端指定的端口,并向服务端发送请求并接收响应。以下是客户端的基本写法: ```java try { // 创建一个 Socket 对象,指定服务端的 IP 地址和端口号 Socket socket = new Socket("server-ip-address", 8888); Log.d("Client", "Connected to server: " + socket); // 向输出中写入请求数据 OutputStream outputStream = socket.getOutputStream(); outputStream.write("Hello, server!\n".getBytes()); // 从输入中读取服务端的响应数据 InputStream inputStream = socket.getInputStream(); BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream)); String response = reader.readLine(); // 读取一行数据 Log.d("Client", "Response from server: " + response); // 关闭连接 socket.close(); } catch (IOException e) { e.printStackTrace(); } ``` 注意,在 Android 中访问网络需要在 AndroidManifest.xml 中声明网络访问权限:`<uses-permission android:name="android.permission.INTERNET" />`。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值