在HarmonyOS NEXT开发中流式布局折叠效果?

在HarmonyOS NEXT开发中流式布局折叠效果?有没有控件,超过2行的不显示,最后一个显示个折叠箭头,点击箭头再展开剩余的行数

本文参与了 【 HarmonyOS NEXT 技术问答冲榜,等你来战!】欢迎正在阅读的你也加入。

阅读 576
avatarAI BotBETA
在HarmonyOS NEXT开发中,要实现流式布局折叠效果,可以使用自定义布局和逻辑来实现。目前HarmonyOS的标准控件库中并没有直接提供这样的折叠效果控件。不过,你可以通过以下步骤来实现:

1. **自定义流式布局**:创建一个自定义的流式布局,用于管理子视图的排列。

2. **行数控制**:在自定义布局中,控制显示的行数。当行数超过2行时,隐藏额外的行,并在最后一行显示一个折叠箭头。

3. **点击事件处理**:为折叠箭头添加点击事件监听器,当点击箭头时,切换显示和隐藏的状态。

4. **动画效果**(可选):为了提升用户体验,可以在展开和折叠时添加动画效果。

你需要编写相应的Java或eTS代码来实现上述逻辑,并结合HarmonyOS的布局和控件系统来完成界面设计。

由于这是一个比较具体的开发需求,没有现成的控件可以直接使用,因此具体的实现细节会依赖于你的应用需求和UI设计。
1 个回答

没有这样的控件,可以使用List实现,具体实现可以参考:https://developer.huawei.com/consumer/cn/doc/harmonyos-refere...
流式布局在HarmonyOS中可以通过flex布局实现,demo:

import measure from '@ohos.measure' 
import display from '@ohos.display'; 
 
const childMaxWidth:number = 325 //为了方便后续计算,这里的宽度数值为px 
let displayClass: display.Display | null = null; 
let componentWidth:number = 0; 
try { 
  displayClass = display.getDefaultDisplaySync(); 
  componentWidth = displayClass.width 
} catch (exception) { 
  console.error('Failed to obtain the default display object. Code: ' + JSON.stringify(exception)); 
} 
@Component 
struct TextItem{ 
  @State message:string = '' 
  @Prop fontSizeData:number 
  build() { 
    Text(this.message) 
      .fontSize(this.fontSizeData) 
      .margin({left:10,top:10}) 
      .backgroundColor('#c4c2cc') 
      .constraintSize({maxWidth:childMaxWidth + 'px'}) 
      .maxLines(1) 
      .textOverflow({overflow:TextOverflow.Ellipsis}) 
  } 
} 
@Entry 
@Component 
struct Index2 { 
  @State @Watch('IndexChange')index:number = 0 
  @State @Watch('textChange') message: string = '' 
  @State FlexWidth: string = '80%'; 
  @State newIndex:number = 0; 
  @State heightControl: number|string = 100; 
  @State ButtonText:string = '∨'; 
  @State AllData:string[] = ['1','22','333','44444','55','666','7777','8888888888888','99','3434','5656','7878','141414141','68681'] 
  @State SomeData:string[] = [] 
  @State ShowData:string[] = [] 
  @State fontSizeData:number = 30 
  @State AllWidth:number = 0 
  @State textWidth:number = 0 
  @State restrictWidth: number = 0; 
  IndexChange(){ 
    if(this.AllWidth >= (this.restrictWidth - childMaxWidth) && this.AllWidth <= (this.restrictWidth)){ 
      this.newIndex = this.index 
      console.log('text1 newIndex',this.newIndex) 
      console.log('text1 change',this.newIndex) 
    } 
  } 
  textChange(){ 
    let content:string = this.message 
    this.textWidth = measure.measureText({ 
      textContent: content, 
      fontSize: this.fontSizeData 
    }) 
    if(this.textWidth > childMaxWidth){ 
      this.AllWidth += childMaxWidth 
    }else{ 
      this.AllWidth += this.textWidth 
    } 
    console.log('text1 content',content) 
    console.log('text1 Width',this.textWidth) 
 
  } 
  aboutToAppear(): void { 
    if(componentWidth != 0){ 
      this.restrictWidth = Math.floor((parseFloat(this.FlexWidth) * componentWidth) * 1.3 * 0.01) 
      console.log('text1 componentWidth',componentWidth) 
      console.log('text1 restrictWidth',this.restrictWidth) 
    } 
    for(let i = 0;i < this.AllData.length;i++){ 
      this.message = this.AllData[i] 
      this.index = i 
    } 
    console.log('text1 change newIndex',this.newIndex) 
    this.SomeData = this.AllData.slice(0,this.newIndex+1) 
    this.ShowData = this.SomeData 
  } 
  build() { 
    Row() { 
      Column() { 
        Flex({wrap:FlexWrap.Wrap}){ 
          ForEach( 
            this.ShowData, 
            (item:string)=>{ 
              TextItem({message:item,fontSizeData:this.fontSizeData}) 
            } 
          ) 
          Button(this.ButtonText) 
            .onClick(()=>{ 
              if(this.heightControl === 100){ 
                this.heightControl = '100%' 
                this.ButtonText = '^' 
                this.ShowData = this.AllData 
              }else{ 
                this.heightControl = 100 
                this.ButtonText = 'v' 
                this.ShowData = this.SomeData 
              } 
            }) 
            .width(40) 
            .height(30) 
            .margin({left:10,top:10}) 
        }.constraintSize({ maxHeight: this.heightControl }) 
        .border({width:1}) 
        .width(this.FlexWidth) 
        .margin({left:'5%'}) 
        .clip(true) 
      } 
      .width('100%') 
    } 
    .height('100%') 
 
  } 
}

本文参与了 【 HarmonyOS NEXT 技术问答冲榜,等你来战!】欢迎正在阅读的你也加入。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进