Android自定義View實現(xiàn)流式布局(熱門標簽效果)

效果圖

實現(xiàn)效果圖

思維導(dǎo)圖

思維導(dǎo)圖

一炸枣、流式布局的實現(xiàn)

實現(xiàn)原理:采用面向?qū)ο笏枷雽⒄麄€布局分為很多行的對象,每個行對象管理自己行內(nèi)的孩子弄唧,這里通過集合來管理适肠。

1. 內(nèi)部類Line的實現(xiàn)

1.1 定義行的基本屬性
  • List<View>:管理行中的孩子

  • maxWidth:行的最大寬度

  • usedWidth:使用的寬度

  • height:行的高度

  • space:孩子之間的間距

  • 構(gòu)造初始化maxWidth和space

      public Line(int maxWidth, int horizontalSpace) {
          this.maxWidth = maxWidth;
          this.space = horizontalSpace;
      }
    
1.2 addView(View view)方法實現(xiàn)
  • 往行的集合里添加View,更新行的使用寬度和高度

      /**
       * 往集合里添加孩子
       */
      public void addView(View view) {
          int childWidth = view.getMeasuredWidth();
          int childHeight = view.getMeasuredHeight();
    
          // 更新行的使用寬度和高度
          if (views.size() == 0) {
              // 集合里沒有孩子的時候
              if (childWidth > maxWidth) {
                  usedWidth = maxWidth;
                  height = childHeight;
              } else {
                  usedWidth = childWidth;
                  height = childHeight;
              }
          } else {
              usedWidth += childWidth + space;
              height = childHeight > height ? childHeight : height;
          }
    
          // 添加孩子到集合
          views.add(view);
      }
    
1.3 canAddView(View view)方法實現(xiàn)
  • 判斷是否能往行里添加孩子候引,如果孩子的寬度大于剩余寬度就不能

      /**
       * 判斷當(dāng)前的行是否能添加孩子
       *
       * @return
       */
      public boolean canAddView(View view) {
          // 集合里沒有數(shù)據(jù)可以添加
          if (views.size() == 0) {
              return true;
          }
    
          // 最后一個孩子的寬度大于剩余寬度就不添加
          if (view.getMeasuredWidth() > (maxWidth - usedWidth - space)) {
              return false;
          }
    
          // 默認可以添加
          return true;
      }
    

2. 對容器進行測量(onMeasure方法的實現(xiàn))

2.1 獲取寬度侯养,計算maxWidth,構(gòu)造傳入Line
  • 總寬度減去左右邊距就是行的最大寬度

      // 獲取總寬度
      int width = MeasureSpec.getSize(widthMeasureSpec);
      // 計算最大的寬度
      mMaxWidth = width - getPaddingLeft() - getPaddingRight();
    
2.2 循環(huán)獲取孩子進行測量
  • 獲取孩子總數(shù)澄干,遍歷獲取每一個孩子沸毁,然后進行測量,測量完之后還需要將孩子添加到行集合里傻寂,然后將行添加到管理行的集合里

      // ******************** 測量孩子 ********************
      // 遍歷獲取孩子
      int childCount = this.getChildCount();
      for (int i = 0; i < childCount; i++) {
          View childView = getChildAt(i);
          // 測量孩子
          measureChild(childView, widthMeasureSpec, heightMeasureSpec);
    
          // 測量完需要將孩子添加到管理行的孩子的集合中息尺,將行添加到管理行的集合中
    
          if (mCurrentLine == null) {
              // 初次添加第一個孩子的時候
              mCurrentLine = new Line(mMaxWidth, HORIZONTAL_SPACE);
    
              // 添加孩子
              mCurrentLine.addView(childView);
              // 添加行
              mLines.add(mCurrentLine);
    
          } else {
              // 行中有孩子的時候,判斷時候能添加
              if (mCurrentLine.canAddView(childView)) {
                  // 繼續(xù)往該行里添加
                  mCurrentLine.addView(childView);
              } else {
                  //  添加到下一行
                  mCurrentLine = new Line(mMaxWidth, HORIZONTAL_SPACE);
                  mCurrentLine.addView(childView);
                  mLines.add(mCurrentLine);
              }
          }
      }
    

2.3 測量自己

  • 由于寬度肯定是填充整個屏幕疾掰,這里只需要處理行的高度搂誉,累加所有的行高和豎直邊距算出高度

      // ******************** 測量自己 *********************
      // 測量自己只需要計算高度,寬度肯定會被填充滿的
      int height = getPaddingTop() + getPaddingBottom();
      for (int i = 0; i < mLines.size(); i++) {
          // 所有行的高度
          height += mLines.get(i).height;
      }
      // 所有豎直的間距
      height += (mLines.size() - 1) * VERTICAL_SPACE;
    
      // 測量
      setMeasuredDimension(width, height);
    

3. 指定孩子的顯示位置(onLayout方法的實現(xiàn))

實現(xiàn)思路:指定孩子的位置静檬,孩子給了行管理炭懊,所以這里具體孩子的位置應(yīng)該交給行去指定。容器只需要指定行的位置就可以拂檩。

  • 遍歷獲取所有的行侮腹,讓行去指定孩子的位置,指定行的高度

      @Override
      protected void onLayout(boolean changed, int l, int t, int r, int b) {
          // 這里只負責(zé)高度的位置稻励,具體的寬度和子孩子的位置讓具體的行去管理
          l = getPaddingLeft();
          t = getPaddingTop();
          for (int i = 0; i < mLines.size(); i++) {
              // 獲取行
              Line line = mLines.get(i);
              // 管理
              line.layout(t, l);
    
              // 更新高度
              t += line.height;
              if (i != mLines.size() - 1) {
                  // 不是最后一條就添加間距
                  t += VERTICAL_SPACE;
              }
          }
      }
    

4. Line中l(wèi)ayout方法的實現(xiàn)(指定孩子的位置)

  • 遍歷獲取每一個孩子父阻,獲取孩子的寬度和高度愈涩,計算上下左右的大小,指定孩子的位置加矛,之后還需要更新孩子左邊的大小

      // 循環(huán)指定孩子位置
      for (View view : views) {
          // 獲取寬高
          int measuredWidth = view.getMeasuredWidth();
          int measuredHeight = view.getMeasuredHeight();
          // 重新測量
          view.measure(MeasureSpec.makeMeasureSpec(measuredWidth + avg, MeasureSpec.EXACTLY),
                  MeasureSpec.makeMeasureSpec(measuredHeight, MeasureSpec.EXACTLY));
          // 重新獲取寬度值
          measuredWidth = view.getMeasuredWidth();
    
          int top = t;
          int left = l;
          int right = measuredWidth + left;
          int bottom = measuredHeight + top;
          // 指定位置
          view.layout(left, top, right, bottom);
    
          // 更新數(shù)據(jù)
          l += measuredWidth + space;
      }
    

5. 細節(jié)處理

  • 第一次測量之后履婉,行管理器中就有了行的對象,之后每次測量都會去創(chuàng)建下一行斟览,這樣就會出現(xiàn)很多空行出來毁腿,所以需要在測量之前將集合清空。

      mLines.clear();
      mCurrentLine = null;
    
  • 每一行的最后一個孩子放不下就放到下一行苛茂,這樣每一行就都會有空格已烤,這里將這些空格平分給行里的每一個孩子,重新指定其寬度妓羊。

      // 平分剩下的空間
      int avg = (maxWidth - usedWidth) / views.size();
    
      // 重新測量
      view.measure(MeasureSpec.makeMeasureSpec(measuredWidth + avg, MeasureSpec.EXACTLY),
              MeasureSpec.makeMeasureSpec(measuredHeight, MeasureSpec.EXACTLY));
      // 重新獲取寬度值
      measuredWidth = view.getMeasuredWidth();
    

6. 使用自定義屬性草戈,將水平間距和豎直間距做成屬性,在布局中指定侍瑟,增強擴展性

  • attrs文件指定屬性名

     <declare-styleable name="FlowLayout">
          <attr name="width_space" format="dimension"/>
          <attr name="height_space" format="dimension"/>
    </declare-styleable>
    
  • 構(gòu)造中獲取屬性

      // 獲取自定義屬性
      TypedArray array = context.obtainStyledAttributes(attrs, R.styleable.FlowLayout);
      horizontal_space = array.getDimension(R.styleable.FlowLayout_width_space,0);
      vertical_space =  array.getDimension(R.styleable.FlowLayout_height_space,0);
      array.recycle();
    
  • 布局中使用屬性

      app:width_space="10dp"
      app:height_space="10dp"
    
經(jīng)過以上步驟之后,F(xiàn)lowLayout基本就已經(jīng)實現(xiàn)了丙猬,接下來就是使用了涨颜。

二、流式布局的使用

  • 布局中申明

      <ScrollView
          xmlns:android="http://schemas.android.com/apk/res/android"
          xmlns:tools="http://schemas.android.com/tools"
          android:layout_width="match_parent"
          android:layout_height="match_parent"
          xmlns:app="http://schemas.android.com/apk/res-auto"
          android:fillViewport="true"
          tools:context="com.pinger.sample.MainActivity">
      
          <com.pinger.library.FlowLayout
              app:width_space="10dp"
              app:height_space="10dp"
              android:id="@+id/flow_layout"
              android:layout_width="match_parent"
              android:layout_height="match_parent"
              android:padding="5dp"/>
      </ScrollView>
    
  • 代碼中使用

  • 其實就是循環(huán)遍歷數(shù)據(jù)的長度茧球,不斷的創(chuàng)建TextView庭瑰,然后設(shè)置TextView的屬性和背景,包括五彩背景等抢埋,最后將TextView添加到FlowLayout中就可以弹灭。


歡迎大家訪問我的簡書博客GitHub揪垄。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末穷吮,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子饥努,更是在濱河造成了極大的恐慌捡鱼,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,602評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件酷愧,死亡現(xiàn)場離奇詭異驾诈,居然都是意外死亡,警方通過查閱死者的電腦和手機溶浴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,442評論 2 382
  • 文/潘曉璐 我一進店門乍迄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人士败,你說我怎么就攤上這事闯两。” “怎么了?”我有些...
    開封第一講書人閱讀 152,878評論 0 344
  • 文/不壞的土叔 我叫張陵生蚁,是天一觀的道長噩翠。 經(jīng)常有香客問我,道長邦投,這世上最難降的妖魔是什么伤锚? 我笑而不...
    開封第一講書人閱讀 55,306評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮志衣,結(jié)果婚禮上屯援,老公的妹妹穿的比我還像新娘。我一直安慰自己念脯,他們只是感情好狞洋,可當(dāng)我...
    茶點故事閱讀 64,330評論 5 373
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著绿店,像睡著了一般吉懊。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上假勿,一...
    開封第一講書人閱讀 49,071評論 1 285
  • 那天借嗽,我揣著相機與錄音,去河邊找鬼转培。 笑死恶导,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的浸须。 我是一名探鬼主播惨寿,決...
    沈念sama閱讀 38,382評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼删窒!你這毒婦竟也來了裂垦?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,006評論 0 259
  • 序言:老撾萬榮一對情侶失蹤肌索,失蹤者是張志新(化名)和其女友劉穎缸废,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體驶社,經(jīng)...
    沈念sama閱讀 43,512評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡企量,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,965評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了亡电。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片届巩。...
    茶點故事閱讀 38,094評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖份乒,靈堂內(nèi)的尸體忽然破棺而出恕汇,到底是詐尸還是另有隱情腕唧,我是刑警寧澤,帶...
    沈念sama閱讀 33,732評論 4 323
  • 正文 年R本政府宣布瘾英,位于F島的核電站枣接,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏缺谴。R本人自食惡果不足惜但惶,卻給世界環(huán)境...
    茶點故事閱讀 39,283評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望湿蛔。 院中可真熱鬧膀曾,春花似錦、人聲如沸阳啥。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,286評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽察迟。三九已至斩狱,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間扎瓶,已是汗流浹背所踊。 一陣腳步聲響...
    開封第一講書人閱讀 31,512評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留栗弟,地道東北人。 一個月前我還...
    沈念sama閱讀 45,536評論 2 354
  • 正文 我出身青樓工闺,卻偏偏與公主長得像乍赫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子陆蟆,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,828評論 2 345

推薦閱讀更多精彩內(nèi)容