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

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

作者: MardaWang | 来源:发表于2024-11-19 20:42 被阅读0次

前一章节,我们完成了欢迎页的逻辑和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组件的特性和使用,下一篇文章再做具体讲解。

相关文章

网友评论

      本文标题:从零开始仿抖音做一个APP(主页面tabs组件)

      本文链接:https://www.haomeiwen.com/subject/dgkpjjtx.html