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

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

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

瀏覽:28日期: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久久,三上悠亚国产精品一区二区三区
国产麻豆精品久久| 亚洲一二三区视频| 噜噜噜久久亚洲精品国产品小说| 欧美日本二区| 国产精品www994| 欧美日韩尤物久久| 日韩成人免费| 成人看片网站| 欧美成人亚洲| 亚洲色图网站| 欧美性www| 美女av一区| 玖玖玖国产精品| 国产欧美自拍| 久久久成人网| 99视频一区| 久久理论电影| 亚洲欧美日韩国产一区二区| 国内亚洲精品| 国产综合激情| 欧美一区久久久| 婷婷成人综合| 亚洲乱码视频| 麻豆极品一区二区三区| 精品欧美一区二区三区在线观看| 国产高清一区| 日韩av资源网| av资源亚洲| 亚洲综合日韩| 国产精品日本一区二区不卡视频| 国产一区2区在线观看| 亚洲国产成人精品女人| 欧美一级网址| 亚洲欧美日本视频在线观看| 亚洲精品一区二区妖精| 国产精品91一区二区三区| 国产日韩中文在线中文字幕| 久久精品高清| 欧美日韩国产观看视频| 国产suv精品一区| 精品国模一区二区三区| 亚洲精品美女91| 日韩一区精品视频| 欧美日韩一区二区三区不卡视频 | 欧美亚洲国产一区| 视频一区二区三区入口| 加勒比视频一区| 视频在线观看一区| 美日韩一区二区三区| 亚洲国产影院| 国产精品一区二区免费福利视频 | 亚洲国产一区二区三区在线播放| 日韩av午夜在线观看| 成人免费一区| 亚洲欧美日韩一区在线观看| 久久av日韩| 欧美亚洲色图校园春色| 亚洲国产日韩欧美在线| 午夜国产精品视频| 欧美成人一二区| 在线一区视频| 精品免费av| 成人影视亚洲图片在线| 国产欧美自拍一区| 国产极品模特精品一二| 国产精品宾馆| 99精品在线| 日韩一区二区三区精品| 青青草91久久久久久久久| 日韩精品久久久久久久电影99爱| 中文视频一区| 日韩中文影院| 国产精品久久亚洲不卡| 男人的天堂亚洲一区| 国产白浆在线免费观看| 日韩av中文字幕一区二区| 久久在线电影| 日韩av在线播放网址| 欧美日韩一区二区三区不卡视频 | 久久国产欧美| 国产精品1区| 日本亚洲欧美天堂免费| 欧美va天堂在线| 国产96在线亚洲| 国产极品一区| 欧美一区自拍| 亚洲精品大片| 国产精品嫩草99av在线| 91精品蜜臀一区二区三区在线 | 国产精品美女在线观看直播 | 精品国内亚洲2022精品成人| 欧美国产91| 国产专区一区| 97精品国产一区二区三区| 日韩视频久久| 91日韩免费| 久久精品一本| 国产精品美女久久久久久不卡| 麻豆久久久久久| 国产精品女主播一区二区三区| 亚洲播播91| 国产精品一区二区精品| 日本不卡一区二区三区| 亚洲ab电影| 亚洲日本国产| 日韩一区免费| 91综合久久爱com| 日韩高清在线观看一区二区| 天使萌一区二区三区免费观看| 午夜免费一区| 欧美日韩精品一区二区视频| 中文字幕在线看片| 日本一区二区免费高清| 久久一区国产| 乱一区二区av| 精品免费av一区二区三区| 国产精品一区二区三区美女| 欧美一区二区三区久久| 国产精品一线| 麻豆国产精品| 成人精品久久| 韩国三级一区| 999精品一区| 亚洲高清不卡| 99国产一区| 丝袜诱惑制服诱惑色一区在线观看| 黄色成人精品网站| 久久最新视频| 日韩欧美2区| 国产欧美日韩一区二区三区在线| 国产区精品区| 国产精选久久| 国产伦精品一区二区三区在线播放| 一本一道久久a久久| 午夜国产一区二区| 国产麻豆精品久久| 国产精品久久久久久久久久白浆 | 久久精品国产99国产精品| 四虎成人精品一区二区免费网站| 欧美日韩高清| 麻豆91精品| 亚洲精品观看| 国产激情久久| 亚洲伦乱视频| 免费日韩av| 日韩高清在线不卡| 国产精品一卡| 日韩精品久久久久久久电影99爱| 精品1区2区3区4区| 日韩免费精品| 国产96在线亚洲| 亚洲免费高清| 国产亚洲一区二区三区啪| 国产极品久久久久久久久波多结野| 精品视频免费| 91精品高清| 日本不卡视频在线观看| 91p九色成人| 精品美女视频| 欧美日韩激情| 日韩精彩视频在线观看| 国产va免费精品观看精品视频| 蜜臀久久99精品久久一区二区| 综合欧美亚洲| 水蜜桃精品av一区二区| 老牛影视一区二区三区| 久久av影院| 91超碰国产精品| 视频一区中文字幕国产| 国产日韩欧美三级| 欧美中文一区二区| 日韩欧美激情| 九九精品调教| 亚洲a级精品| 91中文字幕精品永久在线| 在线成人直播| 欧美a级一区二区| 伊人影院久久| 欧美激情99| 妖精视频成人观看www| 国产精品伊人| 制服诱惑一区二区| 老司机精品视频网| 亚洲欧美不卡| 日本黄色精品| 日韩va欧美va亚洲va久久| 久久精品电影| 国产极品模特精品一二| 亚洲精品a级片| 国产在线观看91一区二区三区| 丝袜亚洲精品中文字幕一区| 91欧美国产| 日韩高清不卡一区| 夜鲁夜鲁夜鲁视频在线播放| 日韩av资源网| 国内精品99| 麻豆国产精品777777在线| 玖玖精品视频| 国产高潮在线| 国产图片一区|