日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Android自定義流式布局的實現示例

瀏覽:30日期:2022-09-21 11:55:15

在日常的app使用中,我們會在android 的app中看見 熱門標簽等自動換行的流式布局,今天,我們就來看看如何自定義一個類似熱門標簽那樣的流式布局。下面我們就來詳細介紹流式布局的應用特點以及用的的技術點。

1.流式布局的特點以及應用場景

特點:當上面一行的空間不夠容納新的TextView時候,才開辟下一行的空間。

原理圖:

Android自定義流式布局的實現示例

場景:主要用于關鍵詞搜索或者熱門標簽等場景

2.自定義ViewGroup

(1)onMeasure:測量子view的寬高,設置自己的寬和高(2)onLayout:設置子view的位置onMeasure:根據子view的布局文件中屬性,來為子view設置測量模式和測量值測量=測量模式+測量值;

測量模式有3種:

EXACTLY:表示設置了精確的值,一般當childView設置其寬、高為精確值、match_parent時,ViewGroup會將其設置為EXACTLY; AT_MOST:表示子布局被限制在一個最大值內,一般當childView設置其寬、高為wrap_content時,ViewGroup會將其設置為AT_MOST; UNSPECIFIED:表示子布局想要多大就多大,一般出現在AadapterView的item的heightMode中、ScrollView的childView的heightMode中;此種模式比較少見。

(3)LayoutParams

ViewGroup LayoutParams :每個 ViewGroup 對應一個 LayoutParams; 即 ViewGroup -> LayoutParams getLayoutParams 不知道轉為哪個對應的LayoutParams ,其實很簡單,就是如下: 子View.getLayoutParams 得到的LayoutParams對應的就是 子View所在的父控件的LayoutParams;

例如,LinearLayout 里面的子view.getLayoutParams ->LinearLayout.LayoutParams。所以 咱們的FlowLayout 也需要一個LayoutParams,由于上面的效果圖是子View的 margin,所以應該使用MarginLayoutParams。即FlowLayout->MarginLayoutParams

3.代碼

FlowLayoutView

/** * Created by zzs on 20/01/16. * 流式布局 */public class FlowLayoutView extends ViewGroup{ public FlowLayoutView(Context context) { super(context); } public FlowLayoutView(Context context, AttributeSet attrs) { super(context, attrs); } //或ViewGroup中所有子元素的屬性 //否則只獲取ViewGroup 的屬性 @Override public LayoutParams generateLayoutParams(AttributeSet attrs) { return new MarginLayoutParams(getContext(),attrs); } /** * 測量控件本身的大小 寬和高 根據子內容獲取 * @param widthMeasureSpec * @param heightMeasureSpec */ @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); //獲取父布局的模式和尺寸 int sizeWidth = MeasureSpec.getSize(widthMeasureSpec); int modeWidth = MeasureSpec.getMode(widthMeasureSpec); int sizeHeight = MeasureSpec.getSize(heightMeasureSpec); int modeHeight = MeasureSpec.getMode(heightMeasureSpec); //記錄Wrap_content的寬高 int width = 0; int height = 0; //沒一行的寬和高 int lineWidth = 0; int lineHeight =0; //獲取子元素的數量 int cCount = getChildCount(); //進行遍歷子元素 for(int i=0;i<cCount;i++){ //獲取每一個子元素 View child = getChildAt(i); //測量每一個子元素 measureChild(child,widthMeasureSpec,heightMeasureSpec); //獲取每個子元素的屬性 MarginLayoutParams layoutParams = (MarginLayoutParams) child.getLayoutParams(); //獲取當前子元素的寬和高 int childWidth = child.getMeasuredWidth()+layoutParams.leftMargin+layoutParams.rightMargin; int childHeight = child.getMeasuredHeight()+layoutParams.bottomMargin+layoutParams.topMargin; if(lineWidth+childWidth>getWidth()-getPaddingLeft()-getPaddingRight()){//換行 width = Math.max(lineWidth,childWidth); lineWidth = childWidth;//重新開一行 當前行的寬度 //疊加之前的高度 height += lineHeight; //重新開一行 這個新行的高度 lineHeight = childHeight; }else{//不換行 //記錄當前行的寬 lineWidth += childWidth; //記錄一行中的高度 lineHeight = Math.max(lineHeight,childHeight); } if(i ==cCount-1){//判斷是否是最后一個元素 width = Math.max(width,lineWidth); height += lineHeight; } //設置ViewGrop最終寬高 setMeasuredDimension(modeWidth==MeasureSpec.EXACTLY?sizeWidth:width+getPaddingRight()+getPaddingLeft(),modeHeight==MeasureSpec.EXACTLY?sizeHeight:height+getPaddingBottom()+getPaddingTop()); } } //記錄所有子元素 按行存取 List<List<View>> mAllViews = new ArrayList<>(); //記錄每一行中最大的高度 private List<Integer> mLineHeight = new ArrayList<>(); //擺放子View ??????????????? @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { //為了多次執行時 數據正確 mAllViews.clear(); mLineHeight.clear(); //獲取父控件的寬度 int width = getWidth(); //記錄一行中的寬和高 int lineWidth = 0; int lineHeight = 0; //記錄一行中的 View List<View> lineView = new ArrayList<>(); //獲取子元素的個數 int cCount = getChildCount(); Log.e('AAA','==AAAA=>'+cCount); //遍歷所有子元素 for(int i=0;i<cCount;i++){ View child = getChildAt(i); //獲取每個子元素的屬性 MarginLayoutParams layoutParams = (MarginLayoutParams) child.getLayoutParams(); //獲取當前子元素的寬和高 int childWidth = child.getMeasuredWidth()+layoutParams.leftMargin+layoutParams.rightMargin; int childHeight = child.getMeasuredHeight()+layoutParams.bottomMargin+layoutParams.topMargin; if(lineWidth+childWidth>width-getPaddingLeft()-getPaddingRight()){//判斷是否換行 //記錄一行的高 和一行中的View mLineHeight.add(lineHeight); mAllViews.add(lineView); Log.e('AAA','==>VVV===>'+mAllViews.size()); //讓數據重置 lineWidth = 0; lineView = new ArrayList<>(); }else{ //記錄一行中的寬 高 view lineWidth += childWidth; lineHeight = Math.max(lineHeight,childHeight); lineView.add(child); } }//處理元素 數據以每行的形式處理 mLineHeight.add(lineHeight); mAllViews.add(lineView); //獲取每個View的padding int left = getPaddingLeft(); int top = getPaddingTop(); //獲取有多少行 int lineNum = mAllViews.size(); Log.e('AAA','=BBB=>'+lineNum); //展示每一行 for(int i=0;i<lineNum;i++){ lineHeight = mLineHeight.get(i); lineView = mAllViews.get(i); //遍歷每一行 // 取出 每一行的 所有的View lineView = mAllViews.get(i); lineHeight = mLineHeight.get(i); // 循環遍歷 每一行的View for(int j=0;j<lineView.size();j++){ View child = lineView.get(j); MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); int lc = left+lp.leftMargin; int tc = top+lp.topMargin; int rc = lc+child.getMeasuredWidth(); int bc = tc+child.getMeasuredHeight(); //擺放 子View的位置 child.layout(lc, tc, rc, bc); //疊加 前一個View 的 實際寬度 left += child.getMeasuredWidth()+lp.leftMargin+lp.rightMargin; } //換行了 重置 left = getPaddingLeft(); top+=lineHeight; } }}

MainActivity

public class MainActivity extends AppCompatActivity { private FlowLayoutView flowLayoutView; private String[] titils = {'sds', 'da', 'aa', 'fff', 'gggggg', 'hhgghg', 'jjhhhh', 'aaaaaaaaaaaaaaaaaaaaaaaaaaa', 'fff', 'gggggg', 'hhgghg', 'jjhhhh', 'sds', 'da', 'aa', 'fff', 'gggggg', 'sds', 'da', 'aa', 'fff', 'gggggg'}; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); flowLayoutView = findViewById(R.id.flowLayoutView); for (int i = 0; i < titils.length; i++) { TextView tv = (TextView) LayoutInflater.from(MainActivity.this) .inflate(R.layout.lv_item, flowLayoutView, false); tv.setText(titils[i]); flowLayoutView.addView(tv); } }}

activity_main

<?xml version='1.0' encoding='utf-8'?><LinearLayout xmlns:android='http://schemas.android.com/apk/res/android' xmlns:app='http://schemas.android.com/apk/res-auto' xmlns:tools='http://schemas.android.com/tools' android:layout_width='match_parent' android:layout_height='match_parent' android:orientation='vertical' tools:context='.MainActivity'> <com.zzs.polygon.FlowLayoutView android: android:layout_width='match_parent' android:layout_height='wrap_content' /></LinearLayout>

lv_item

<?xml version='1.0' encoding='utf-8'?><TextView xmlns:android='http://schemas.android.com/apk/res/android' android:layout_width='wrap_content' android:layout_height='wrap_content' android:background='@drawable/tv_shape' android:paddingLeft='10dp' android:paddingTop='5dp' android:paddingRight='10dp' android:paddingBottom='5dp' android:layout_marginLeft='10dp' android:layout_marginBottom='5dp' android:layout_marginRight='10dp' android:layout_marginTop='5dp' android:text='AAA' android:textColor='#ff0000'></TextView>

drawable / tv_shape.xml

<?xml version='1.0' encoding='utf-8'?><shape xmlns:android='http://schemas.android.com/apk/res/android'> <corners android:radius='30dp'></corners> <solid android:color='#8800ff00'></solid> <padding android:left='10dp' android:right='10dp' android:top='10dp' android:bottom='10dp' ></padding></shape></TextView>

效果圖

Android自定義流式布局的實現示例

到此這篇關于Android自定義流式布局的實現示例的文章就介紹到這了,更多相關Android 自定義流式布局內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
四虎国产精品免费观看| 欧美日韩国产在线一区| 午夜性色一区二区三区免费视频| 中文字幕高清在线播放| 蜜臀国产一区| 日韩免费视频| 啪啪国产精品| 午夜久久福利| 影音国产精品| 六月丁香综合| 日韩成人一级| 久久精品一区二区三区中文字幕| 精品九九久久| 欧美 日韩 国产一区二区在线视频| 欧美日韩激情在线一区二区三区| 狠狠爱www人成狠狠爱综合网| 天使萌一区二区三区免费观看| 免费精品视频在线| 午夜精品福利影院| 麻豆精品蜜桃视频网站| 伊人网在线播放| 亚洲欧洲一区二区天堂久久| 亚洲精品无吗| 麻豆精品99| 91精品国产乱码久久久久久久| 性色一区二区| 国产精品一区二区美女视频免费看 | 日韩精品久久理论片| 国产欧美日韩一区二区三区四区 | 国产一区二区三区黄网站| 日本精品在线中文字幕| 日韩在线a电影| 国产精品一区二区三区av| 激情黄产视频在线免费观看| 一区久久精品| 国产日韩一区二区三区在线播放| 91中文字幕精品永久在线| 欧美天堂亚洲电影院在线观看| 啪啪亚洲精品| 日本高清不卡一区二区三区视频| 麻豆精品91| 欧美国产极品| 狠狠色综合网| 国产免费av一区二区三区| 日韩欧美精品一区| 亚洲久久视频| 97精品国产| 玖玖玖国产精品| 精品欧美视频| 亚洲一区二区日韩| 精品国产欧美日韩一区二区三区| 亚洲国产日韩欧美在线| 国产欧美日韩一区二区三区在线| 日本欧美不卡| 日韩高清成人在线| 久久精品亚洲人成影院| 日韩av资源网| 久久精品国产99久久| 日韩欧美中文字幕一区二区三区| 国产一区二区三区四区大秀| 蜜桃久久av| 成人片免费看| 欧美一区不卡| 午夜欧美精品久久久久久久| 乱一区二区av| 蜜桃一区二区三区在线观看| 四虎国产精品免费观看| 日韩精选在线| 在线看片不卡| 日韩av在线播放网址| 亚洲精品影视| 欧美福利专区| 国产精品福利在线观看播放| 日韩精品中文字幕一区二区| 99久久精品费精品国产| 国产精品白浆| 亚洲资源网站| 91精品一区二区三区综合| 国产欧美在线观看免费| 亚洲尤物在线| 精品日韩视频| 久久精品国产免费| 欧美一区激情| 深夜日韩欧美| 久久都是精品| 黑丝一区二区| 久久人人97超碰国产公开结果| 美女在线视频一区| 日本电影久久久| 尤物在线精品| 久久免费国产| 久久久久免费| 久久99蜜桃| 国产乱码精品一区二区三区亚洲人| 免播放器亚洲| 宅男在线一区| 欧美日韩尤物久久| 欧美激情国产在线| 精品99在线| 卡一精品卡二卡三网站乱码| 国产精品色婷婷在线观看| 日本不卡高清| 日韩综合一区二区| 老色鬼久久亚洲一区二区| 国产一区观看| 国户精品久久久久久久久久久不卡| 国产aⅴ精品一区二区三区久久 | 国产精品日本| 国内激情久久| 欧美亚洲国产激情| 蜜桃精品在线| 伊人久久高清| 久久久777| 久久久一二三| 九色porny丨国产首页在线| 精品久久在线| 国产一区二区三区四区二区| 麻豆精品一区二区综合av| 国产精品**亚洲精品| 国产精品美女午夜爽爽| 国产精品午夜一区二区三区| 亚洲精品在线二区| 日本欧美韩国一区三区| 911精品国产| 国产日本精品| 国产精品激情| 国产精品99久久免费| 久久的色偷偷| 精品国产午夜| 日本综合字幕| 欧美 日韩 国产一区二区在线视频 | 国产精品xxx在线观看| 欧美国产三级| 国产一区二区三区四区| а√天堂中文在线资源8| 欧美aa一级| 午夜精品一区二区三区国产| 老鸭窝毛片一区二区三区| 亚洲精品观看| 国产精品大片| av资源中文在线| 中文字幕高清在线播放| 一区二区三区视频免费观看 | 日韩一区二区三区在线免费观看| 中文另类视频| 免费久久久久久久久| 久久aⅴ国产紧身牛仔裤| 91欧美极品| 国产不卡精品| 狠狠久久婷婷| 日韩精品视频中文字幕| 国产精品欧美三级在线观看| 国产在视频一区二区三区吞精| 啪啪国产精品| 综合一区二区三区| 麻豆一区二区在线| 99精品国产一区二区三区| 欧美福利在线| 亚洲午夜久久| 欧美国产中文高清| 久久中文视频| 婷婷综合一区| 日韩激情啪啪| 激情国产在线| 日韩在线一区二区| 欧美激情精品| 亚洲精品一区二区妖精| 久久精品凹凸全集| 欧美香蕉视频| 日本亚洲视频在线| 日本不卡免费高清视频在线| 免费视频久久| 久久久精品区| 不卡一区2区| 国产精品宾馆| 一区在线免费| 久久97视频| 麻豆亚洲精品| 精品一区二区三区亚洲| 欧美日韩黑人| 久久影院一区二区三区| 日韩一区二区免费看| 国产精品日本一区二区不卡视频 | 亚洲一区二区毛片| 国产精品久久久久久模特| 亚洲91视频| 国产精品夜夜夜| 亚洲中字黄色| 欧美aⅴ一区二区三区视频| 99热免费精品| 精品一区91| 一二三区精品| 日韩在线视频精品| 国产伦精品一区二区三区视频 | 日本精品一区二区三区在线观看视频| 国产一区二区视频在线看| 亚洲精品极品| 激情婷婷久久| 国产+成+人+亚洲欧洲在线| 蜜臀国产一区二区三区在线播放|