鸿蒙开发实战案例--朋友圈(附源码)_鸿蒙项目源码,面试进阶100题网盘

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新HarmonyOS鸿蒙全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img

img
img
htt

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上鸿蒙开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip204888 (备注鸿蒙)
img

正文

1、头像昵称的叠加布局

2、导航栏的隐藏和显示

3、评论弹窗的动画效果和单例显示

4、点击图片预览

在开始构建页面之前,先解决一下数据问题,我先定义一个数据模型,然后自己手写几条朋友圈数据:

class MomentClass {
public nickName: string; //昵称
public content: string; //内容
public date: string; //发表日期
public showC: boolean; //是否显示评论和赞弹窗
public images: string[]; //内容图片列表

constructor(nickName: string,content: string,date: string,showC: boolean,images:string[]) {
this.nickName = nickName;
this.content = content;
this.date = date;
this.showC = showC;
this.images = images;
}
}

@State momentList: MomentClass[] = [
new MomentClass(‘老张1’,‘今朝有酒今朝醉’,‘8天前’,false,[‘/image/moment_1.png’]),
new MomentClass(‘老张2’,‘hahahah2’,‘8天前’,false,[‘/image/grid_0.png’,‘/image/grid_1.png’,‘/image/grid_2.png’,‘/image/grid_1.png’]),
new MomentClass(‘老张3’,‘哈哈真的很不错’,‘8天前’,false,[]),
new MomentClass(‘老张4’,‘我emo了我emo了我emo了我emo了。。。。。。。。’,‘8天前’,false,[]),
new MomentClass(‘老张5’,‘今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉今朝有酒今朝醉’,‘8天前’,false,[]),
];

有了数据就可以开始开发页面了。我们先忽略导航栏,先看滚动区域部分,显然它是一个List,根据内容不同我把它分成两个ListItemGroup:

第1部分,叠加布局,背景图片上叠加昵称和头像,对齐格式设置bottomEnd,然后调整一下外边距margin就会有底部超出范围的效果:

ListItemGroup(){
ListItem(){
Stack({alignContent:Alignment.BottomEnd}){
Flex() {
Image(‘/image/backImage.png’)
.width(‘100%’)
.height(300)
}

Flex({direction:FlexDirection.Row,justifyContent:FlexAlign.End,alignItems:ItemAlign.Center}){
Text(“幽蓝君”)
.fontColor(Color.White)
.margin({right:5})
Image(‘/image/head.png’)
.width(70)
.height(70)
.borderRadius(8)
}
.padding({right:10})
.width(‘100%’)
.height(70)
.margin({bottom:-20})
}
.width(‘100%’)
.height(300)
}
}

下面的内容区域需要好好分析一下:

图中标出1、2、3部分相信对大家没什么难度,第4部分我使用的是Grid容器,然后根据图片的数量调整列数和行数:代码如下:

Grid(){
ForEach(this.momentItem.images, (imagePath: string,index:number) => {
GridItem() {
Image(imagePath)
}
})
}
.columnsTemplate(this.calcColumnsTemplate())
.rowsTemplate(this.calcRowsTemplate())
.width(this.calcGridRow())
.height(this.calcGridColum())
.columnsGap(7)
.rowsGap(7)
.margin({top:5})

//计算列数
calcRowsTemplate(){
if(this.momentItem.images.length <= 3){
return ‘1fr’
}else if(this.momentItem.images.length > 3 && this.momentItem.images.length <= 6){
return ‘1fr 1fr’
}else {
return ‘1fr 1fr 1fr’
}
}
//计算行数
calcColumnsTemplate(){
if(this.momentItem.images.length <= 1){
return ‘1fr’
}else if(this.momentItem.images.length == 2){
return ‘1fr 1fr’
}else {
return ‘1fr 1fr 1fr’
}
}

第5部分看似是横向布局,它其实是一个叠加布局,黑色弹窗和两侧内容的叠加。代码如下:

Stack({alignContent:Alignment.End}){
Flex({direction:FlexDirection.Row,justifyContent:FlexAlign.SpaceBetween,alignItems:ItemAlign.Center}){
Text(this.momentItem.date)
.fontColor(‘rgb(185,185,185)’)
.fontSize(13)
Text(‘••’)
.width(35)
.height(18)
.borderRadius(5)
.fontSize(12)
.fontWeight(900)
.textAlign(TextAlign.Center)
.fontColor(‘rgb(90,109,150)’)
.backgroundColor(‘rgb(247,247,247)’)

}
.padding({right:20})
.height(40)

Flex({direction:FlexDirection.Row,}){
Flex({direction:FlexDirection.Row}){
Flex({direction:FlexDirection.Row,justifyContent:FlexAlign.Center,alignItems:ItemAlign.Center}){
Image(KaTeX parse error: Expected 'EOF', got '}' at position 122: …tAlign.Center) }̲ .width(60) .he…r(‘app.media.commen’))
.width(18)
.height(18)
Text(‘评论’)
.fontColor(Color.White)
.fontSize(14)
.textAlign(TextAlign.Center)
}
.width(60)
.height(40)
}
.borderRadius(4)
.width(120)
.height(40)
.position({x:0})
.backgroundColor(‘rgb(22,22,22)’)

}
.margin({right:60})
.width(120)
.height(40)

}
.height(40)

这样做完之后,看下效果:

问题来了,评论和赞的弹窗不应该每一行都显示,它只能点击的时候显示一个并且带有动画效果,所以现在全部把它们隐藏,注意这里的隐藏不能使用Visibility.Hidden,而是设置位置和尺寸。因为只有宽度和x坐标的同时变化才能做到朋友圈那样的效果,假设弹窗显示时的x坐标是0,宽度120,那么动画开始之前x初始值应该是120,宽度为0才对,但是幽蓝君亲身实验宽度为0时是不行的,所以这里宽度初始值设置0.01,肉眼看不见就好。

说了一通,不知道大家能不能理解,看代码吧,当右侧的按钮点击时:

@State menuWidth:number = 0.001 //弹窗width初始值
@State menuX:number = 120 //弹窗x值初始值

animateTo({
duration: 300,
onFinish:() => {
this.momentItem.showC = !this.momentItem.showC
}
}, () => {
if(!this.momentItem.showC){
//弹出
this.menuX = 0
this.menuWidth = 120
}else {
//收起
this.menuX = 120
this.menuWidth = 0.001
}
})

这样可以实现单个弹窗的弹出和收起,但是我们要控制整个页面只能出现一个弹窗,所以在上述点击事件中要同时把点击的序号index传给父组件,让父组件处理数据列表,除index外其他行的showC置为false。

问题又来了,我们之前说过,@state修饰的变量,嵌套数据的变化它是监测不到的,所以即便把数据修改了其他弹窗也不会隐藏。

所以子组件中数据对象使用@ObjectLink修饰,数据模型MomentClass也要使用@Observed修饰:

@Observed
class MomentClass {
public nickName: string;
public content: string;
public date: string;
public showC: boolean;
public images: string[];

constructor(nickName: string,content: string,date: string,showC: boolean,images:string[]) {
this.nickName = nickName;
this.content = content;
this.date = date;
this.showC = showC;
this.images = images;
}
}

@ObjectLink @Watch(‘onCountUpdated2’) momentItem: MomentClass;

这样就完成了整个评论弹窗的控制。

接下来就是导航栏了,导航栏有一个显示和隐藏的切换,规律是当第一个listGroupItem滑动消失时导航栏显示,否则隐藏。List组件有一个onScrollIndex方法:

onScrollIndex((start,end)=>{
if(start == 0 && end == 1){
this.navHide = true
}
if(start == 1 && end == 1){
this.navHide = false
}
})

然后把navHide变量赋值给导航栏的hideTitleBar:

hideTitleBar(this.navHide)

这样导航栏就可以按照我们要求显示和隐藏了,至于隐藏时的导航栏图标大家可以自己添加一个,和导航栏反向显示就行了。

最后就是点击预览朋友圈图片,幽蓝君的做法是点击图片弹出一个List组件覆盖到屏幕上,看代码:

if (this.previewAlpha) {
List({initialIndex:this.bigImageIndex}){
ForEach(this.bigImageList,(imagePath)=>{
ListItem(){

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注鸿蒙)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

ndex:this.bigImageIndex}){
ForEach(this.bigImageList,(imagePath)=>{
ListItem(){

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注鸿蒙)
[外链图片转存中…(img-ROL2APnB-1713220782496)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 28
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值