从零开始仿抖音做一个APP(主页面tabs组件)

前一章节,我们完成了欢迎页的逻辑和UI,沉浸式效果也做了讲解和实现,本文简单介绍一下主页面Tabs组件的应用。

Tabs组件介绍

Tabs组件是当前所有应用中最常用的容器组件之一,用户可以在一个页面内快速实现视图内容的切换,极大地提高效率。

Tabs组件主要包含两个部分:TabContent和TabBar。TabContent是内容页,TabBar是导航页签栏,页面结构如下图所示,根据不同的导航类型,布局会有区别,可以分为底部导航、顶部导航、侧边导航,其导航栏分别位于底部、顶部和侧边。


image.png

Tabs组件的写法通常如下所示:

Tabs() {
  TabContent() {
    Text('首页的内容').fontSize(30)
  }
  .tabBar('首页')

  TabContent() {
    Text('推荐的内容').fontSize(30)
  }
  .tabBar('推荐')
  
  TabContent() {
    Text('我的内容').fontSize(30)
  }
  .tabBar("我的")
}

导航栏位置使用Tabs的barPosition参数进行设置。默认情况下,导航栏位于顶部,此时,barPosition为BarPosition.Start。设置为底部导航时,需要将barPosition设置为BarPosition.End。

Tabs({ barPosition: BarPosition.End }) {
  // TabContent的内容:首页、发现、推荐、我的
  ...
}

关于Tabs组件的更多用法,请见官方文档Tabs组件

Tabs组件实现

关于我的tabs实现,直接上代码吧。

定义一个tabs数组(此处第三项是icon资源)

export class AppConstants {
    ......
  /**
   * Tab array.
   */
  static readonly TAB_ARRAY: string[] = [
    'app.string.tab_homepage',
    'app.string.tab_friend',
    'app.media.icon_add',
    'app.string.tab_msg',
    'app.string.tab_mine'
  ];
}

在build绘制

  build() {
    NavDestination() {
      Column() {
        Tabs({ index: this.curTabIndex, controller: this.tabsController }) {
          TabContent() {
            HomePage()
          }
          .width(AppConstants.FULL_PERCENT)
          .height($r('app.string.content_height'))

          TabContent() {
            Text($r(this.tabArray[1]))
              .fontSize($r('app.integer.text_size_20'))
          }
          .width(AppConstants.FULL_PERCENT)
          .height($r('app.string.content_height'))

          TabContent() {
            Text('发布')
              .fontSize($r('app.integer.text_size_20'))
          }
          .width(AppConstants.FULL_PERCENT)
          .height($r('app.string.content_height'))

          TabContent() {
            Text($r(this.tabArray[3]))
              .fontSize($r('app.integer.text_size_20'))
          }
          .width(AppConstants.FULL_PERCENT)
          .height($r('app.string.content_height'))

          TabContent() {
            Text($r(this.tabArray[4]))
              .fontSize($r('app.integer.text_size_20'))
          }
          .width(AppConstants.FULL_PERCENT)
          .height($r('app.string.content_height'))
        }
        .scrollable(false)
        // .barHeight($r('app.integer.tabs_bar_height'))
        // .animationDuration(TabContentConstants.TAB_TABS_DURATION)
        .onChange((index: number) => {
          this.curTabIndex = index;
        })

        Row() {
          ForEach(this.tabArray, (item: string, index: number) => {
            Column() {
              if (index == 2) {
                if (this.curTabIndex == index) {
                  Image($r(item)).width(28).height(24).objectFit(ImageFit.Fill).borderRadius(2)
                } else {
                  Image($r(item)).width(24).height(21).objectFit(ImageFit.Fill).borderRadius(2)
                }
              } else {
                Text($r(item))
                  .fontColor(this.curTabIndex === index ? '#ffffff' : '#66ffffff')
                  .margin({ top: 4 })
              }
            }
            .width(50)
            .margin(10)
            .height(60)
            .onClick(() => {
              this.curTabIndex = index;
              this.tabsController.changeIndex(this.curTabIndex);
            })
          }, (item: string, index: number) => JSON.stringify(item) + index)
        }
        .offset({
          y: $r('app.integer.tab_row_offset')
        })
        .width(AppConstants.FULL_PERCENT)
        // Extend to all non-secure areas.
        .expandSafeArea([SafeAreaType.SYSTEM], [SafeAreaEdge.BOTTOM])
        .backgroundColor($r('app.color.tab_black'))
        .justifyContent(FlexAlign.SpaceAround)
      }
    }
    .hideTitleBar(true)
    .onReady((context: NavDestinationContext) => {
      this.pathStack = context.pathStack
      LogUtils.info("current page config info is :" , JSON.stringify(context.getConfigInRouteMap()));
    })
  }

NavDestination组件从API Version 11开始默认支持安全区避让特性(默认值为:expandSafeArea([SafeAreaType.SYSTEM, SafeAreaType.KEYBOARD, SafeAreaType.CUTOUT], [SafeAreaEdge.TOP, SafeAreaEdge.BOTTOM])),开发者可以重写该属性覆盖默认行为,API Version 11之前的版本需配合expandSafeArea属性实现安全区避让。

以上代码中把tabs标签和tabcontent的绘制实现了抽离,其中tabs角标为0的页面content引用的是home模块的自定义view,大家可以自定义view名称和UI,前文已讲过导出UI或接口的方法,此处不再赘述。

此外,本文在build内部包含了一层NavDestination组件,这是因为前面几篇文章中关于页面跳转,我用的是router跳转,由于官方不再推荐使用router,我这边已改用Navigation组件,大家如果还使用的router,把外面包的这层NavDestination去掉即可。

Navigation作为当前HarmonyOS NEXT开发中主推的路由组件,后续将会频繁被使用到。关于Navigation组件的特性和使用,下一篇文章再做具体讲解。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,137评论 6 511
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,824评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,465评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,131评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,140评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,895评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,535评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,435评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,952评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,081评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,210评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,896评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,552评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,089评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,198评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,531评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,209评论 2 357

推荐阅读更多精彩内容