Android--根據子控制項的大小自動換行的ViewGroup

来源:http://www.cnblogs.com/819158327fan/archive/2016/02/26/5222039.html
-Advertisement-
Play Games

1、自定義ViewGroup 1 /** 2 * Created by Administrator on 2016/2/26. 3 * 4 * --------自動換行的ViewGroup----------- 5 */ 6 public class LineWrapLayout extends V


1、自定義ViewGroup

  1 /**
  2  * Created by Administrator on 2016/2/26.
  3  *
  4  * --------自動換行的ViewGroup-----------
  5  */
  6 public class LineWrapLayout extends ViewGroup {
  7     private static final boolean DEBUG = true;
  8     private static final String TAG = "AutoLineFeedLayout";
  9 
 10     /**
 11      * 左間距
 12      */
 13     private int paddingLeft = 10;
 14     /**
 15      * 右間距
 16      */
 17     private int paddingRight = 10;
 18     /**
 19      *
 20      */
 21     private int paddingTop = 10;
 22     /**
 23      *
 24      */
 25     private int paddingBottom = 10;
 26 
 27     /**
 28      * 水平方向間距
 29      */
 30     private int horizontalSpace = 10;
 31     /**
 32      * 行間距
 33      */
 34     private int verticalSpace = 5;
 35 
 36 
 37     private List<Integer> listX;
 38     private List<Integer> listY;
 39 
 40     public LineWrapLayout(Context context) {
 41         super(context);
 42 
 43     }
 44     public LineWrapLayout(Context context, AttributeSet attrs) {
 45         super(context, attrs);
 46         init(attrs);
 47     }
 48 
 49     public LineWrapLayout(Context context, AttributeSet attrs, int defStyle) {
 50         super(context, attrs, defStyle);
 51         init(attrs);
 52     }
 53 
 54 
 55     @Override
 56     protected void onLayout(boolean changed, int l, int t, int r, int b) {
 57         if(DEBUG) Log.d(TAG, "--- onLayout changed :" + changed + " l :" + l + ",t :" + t + ",r :" + r + ",b :" + b);
 58         int count = getChildCount();
 59         int width = getWidth();
 60         Log.i(TAG, "寬度 :"+width);
 61 
 62 
 63         int startOffsetX = paddingLeft;// 橫坐標開始
 64         int startOffsety = 0;//縱坐標開始
 65         int rowCount = 1;
 66 
 67         int preEndOffsetX = startOffsetX;
 68 
 69         for (int i = 0; i < count; i++) {
 70             final View childView = getChildAt(i);
 71 
 72             int w = childView.getMeasuredWidth();
 73             int h = childView.getMeasuredHeight();
 74 
 75             int x = listX.get(i);
 76             int y = listY.get(i);
 77 
 78             // 佈局子控制項
 79             childView.layout(x, y, x + w, y + h);
 80         }
 81     }
 82     @Override
 83     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
 84         if(DEBUG) Log.v(TAG, "--- onMeasure()");
 85 
 86         int count = getChildCount();
 87         int width = measureWidth(widthMeasureSpec);
 88         Log.i(TAG, "寬度 :"+width);
 89 
 90 
 91         int startOffsetX = paddingLeft;// 橫坐標開始
 92         int startOffsety = 0+paddingTop;//縱坐標開始
 93         int rowCount = 1;
 94 
 95         int preEndOffsetX = startOffsetX;
 96 
 97         listX.clear();
 98         listY.clear();
 99         for (int i = 0; i < count; i++) {
100             Log.v(TAG, "----");
101             final View childView = getChildAt(i);
102             // 設置子空間Child的寬高
103             childView.measure(0,0);
104             /* 獲取子控制項Child的寬高 */
105             int childWidth = childView.getMeasuredWidth();
106             int childHeight = childView.getMeasuredHeight();
107             Log.v(TAG, "childWidth :"+childWidth+" childHeight :"+childHeight);
108             preEndOffsetX = startOffsetX + childWidth /*+ CHILD_MARGIN*/;
109             //TODO [yaojian]margin屬性?
110             if (preEndOffsetX > width - paddingRight ) {
111                 if (startOffsetX > paddingLeft) {
112                     /* 換行  */
113                     startOffsetX = paddingLeft;
114                     startOffsety += childHeight+verticalSpace;
115                     rowCount++;
116                 }
117             }
118             Log.d(TAG, "measure child :"+startOffsetX+", "+startOffsety+", "+preEndOffsetX+", "+(startOffsety+childHeight));
119             listX.add(startOffsetX);
120             listY.add(startOffsety);
121 
122 //            childView.layout(startOffsetX, startOffsety, preEndOffsetX, startOffsety+childHeight);
123             startOffsetX = startOffsetX + childWidth + horizontalSpace;
124         }
125         int lastLineHeight = 0;
126         View lastChild = getChildAt(count-1);
127         if(null != lastChild){
128             lastLineHeight = lastChild.getMeasuredHeight();
129         }
130         setMeasuredDimension(measureWidth(widthMeasureSpec), startOffsety+lastLineHeight+paddingBottom);
131 //        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
132 
133         // 註意setMeasuredDimension和resolveSize的用法
134 //        setMeasuredDimension(resolveSize(measuredWidth, widthMeasureSpec),
135 //                resolveSize(top, heightMeasureSpec));
136     }
137 
138     private int measureWidth(int measureSpec) {
139         int specMode = MeasureSpec.getMode(measureSpec);
140         int specSize = MeasureSpec.getSize(measureSpec);
141 
142 
143         // Default size if no limits are specified.
144         int result = 400;
145 
146         if (specMode == MeasureSpec.AT_MOST) {
147             // Calculate the ideal size of your control
148             // within this maximum size.
149             // If your control fills the available space
150             // return the outer bound.
151             result = specSize;
152         } else if (specMode == MeasureSpec.EXACTLY) {
153             // If your control can fit within these bounds return that value.
154             result = specSize;
155         }
156         return result;
157     }
158 
159     private void init(AttributeSet attrs) {
160         TypedArray attrArray = getContext().obtainStyledAttributes(attrs, R.styleable.AutoLineFeedLayout);
161         int attrCount = attrArray.getIndexCount();
162         for (int i = 0; i < attrCount; i++) {
163             int attrId = attrArray.getIndex(i);
164             switch (attrId) {
165                 case R.styleable.AutoLineFeedLayout_horizontalSpacing:{
166                     float dimen = attrArray.getDimension(attrId, 0);
167                     horizontalSpace = (int) dimen;
168                 }
169                 break;
170                 case R.styleable.AutoLineFeedLayout_verticalSpacing:{
171                     float dimen = attrArray.getDimension(attrId, 0);
172                     verticalSpace = (int) dimen;
173                 }
174                 break;
175                 case R.styleable.AutoLineFeedLayout_paddingBottom:{
176                     float dimen = attrArray.getDimension(attrId, 0);
177                     paddingBottom = (int) dimen;
178                 }
179                 break;
180                 case R.styleable.AutoLineFeedLayout_paddingLeft:{
181                     float dimen = attrArray.getDimension(attrId, 0);
182                     paddingLeft = (int) dimen;
183                 }
184                 break;
185                 case R.styleable.AutoLineFeedLayout_paddingRight:{
186                     float dimen = attrArray.getDimension(attrId, 0);
187                     paddingRight = (int) dimen;
188                 }
189                 break;
190                 case R.styleable.AutoLineFeedLayout_paddingTop:{
191                     float dimen = attrArray.getDimension(attrId, 0);
192                     paddingTop = (int) dimen;
193                 }
194                 break;
195                 case R.styleable.AutoLineFeedLayout_debug:{
196 
197                 }
198                 break;
199 
200                 default:
201                     break;
202             }
203 
204         }
205 
206         listX = new ArrayList<Integer>();
207         listY = new ArrayList<Integer>();
208     }
209 }

2、有一部分自定義屬性可供使用 attrs.xml

    <declare-styleable name="AutoLineFeedLayout">
        <attr name="debug" format="boolean"></attr>

        <attr name="paddingLeft" format="reference|dimension"/>
        <attr name="paddingRight" format="reference|dimension"/>
        <attr name="paddingTop" format="reference|dimension"/>
        <attr name="paddingBottom" format="reference|dimension"/>

        <attr name="verticalSpacing" format="reference|dimension"/>
        <attr name="horizontalSpacing" format="reference|dimension"/>
    </declare-styleable>

3、在佈局文件中和普通ViewGroup使用相同

4、在Activity中動態添加View或者ViewGroup

for (int i = 0;i < myData.getList().length;i++){
                        View child = View.inflate(ZhuanTiActivity.this,R.layout.item_mygridview_layout,null);
                        TextView textView = (TextView) child.findViewById(R.id.tv_title_item);
                        textView.setText(myData.getList()[i].getName());
                        textView.setTextColor(Color.argb(255, colorR, colorG, colorB));
                        textView.setOnClickListener(new View.OnClickListener() {
                            @Override
                            public void onClick(View v) {
                                Log.d("aaa",textView.getText());
                            }
                        });
                        custom_index_zhuanTiActivity.addView(child);
                    }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 這是因為 app.use(function * (){ 語句中有一個 * ,這種方式被稱為generator functions ,一般寫作function *(){...} 的形式,在此類function 中可以支持ES6的一種yield概念。 為了保證這種新型的方法可以編譯通過,在運行node 
  • This application's application-identifier entitlement does not match that of the installed application. These values must match for an upgrade to be a
  • 第一步、首先在你項目中創建一個包存放支持下拉刷新和上拉載入的類: 第二步、需要把兩個動畫導入進來,實現180度旋轉與360度旋轉: 第三步、需要把支持的下拉與上拉顯示的隱藏載入佈局給導入進來 第四步、需要添加strings.xml與colors.xml文件的內容添加到項目裡面: strings.xm
  • 分類:C#、Android、VS2015; 創建日期:2016-02-27 一、簡介 這一節演示如何利用以非同步方式(async、await)訪問SQLite資料庫。 二、示例4運行截圖 下麵左圖為初始頁面,右圖為單擊【創建資料庫】按鈕後的結果。 下麵左圖為單擊【添加單行】按鈕的結果,右圖為單擊【添加...
  • 近期因為項目的需要,研究了一下Android文件下載進度顯示的功能實現,接下來就和大家一起分享學習一下,希望對廣大初學者有幫助。 先上效果圖: 上方的藍色進度條,會根據文件下載量的百分比進行載入,中部的文本控制項用來現在文件下載的百分比,最下方的ImageView用來展示下載好的文件,項目的目的就是動
  • 作者: "@gzdaijie" 本文為作者原創,轉載請註明出處:http://www.cnblogs.com/gzdaijie/p/5222191.html 1.寫在前面 Android提供了豐富的 函數,本文介紹最常用的8種對話框的使用方法,包括普通(包含提示消息和按鈕)、列表、單選、多選、等待、
  • 從啟動界面到主界面之後的效果如圖所示,採用的是v4包下的DrawerLayout, activity_main.xml文件如下: ...
  • ReactNative高級交流群 127482131 或訪問 http://blog.1ygowu.com ReactNative技術專題
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...