【引言】(完整代码在最后面)

本文将通过一个简单的计数器应用案例,介绍如何利用鸿蒙NEXT的特性开发高效、美观的应用程序。我们将涵盖计数器的基本功能实现、用户界面设计、数据持久化及动画效果的添加。

【环境准备】

电脑系统:windows 10

开发工具:DevEco Studio 5.0.1 Beta3 Build Version: 5.0.5.200

工程版本:API 13

真机:Mate60 Pro

语言:ArkTS、ArkUI

【项目概述】

本项目旨在创建一个多计数器应用,用户可以自由地添加、编辑、重置和删除计数器。每个计数器具有独立的名称、当前值、增加步长和减少步长。应用还包括总计数的显示,以便用户快速了解所有计数器的总和。

【功能实现】

1、计数器模型

首先,我们定义了一个CounterItem类来表示单个计数器,其中包含了计数器的基本属性和行为。

@ObservedV2
class CounterItem {
id: number = ++Index.counterId;
@Trace name: string;
@Trace count: number = 0;
@Trace scale: ScaleOptions = { x: 1, y: 1 };
upStep: number = 1;
downStep: number = 1; constructor(name: string) {
this.name = name;
}
}

2、应用入口与状态管理

应用的主入口组件Index负责管理计数器列表、总计数、以及UI的状态。这里使用了@State和@Watch装饰器来监控状态的变化。

@Entry
@Component
struct Index {
static counterStorageKey: string = "counterStorageKey";
static counterId: number = 0; @State listSpacing: number = 20;
@State listItemHeight: number = 120;
@State baseFontSize: number = 60;
@State @Watch('updateTotalCount') counters: CounterItem[] = [];
@State totalCount: number = 0;
@State isSheetVisible: boolean = false;
@State selectedIndex: number = 0; // ...其他方法
}

3、数据持久化

为了保证数据在应用重启后仍然可用,我们使用了preferences模块来同步地读取和写入数据。

saveDataToLocal() {
const saveData: object[] = this.counters.map(counter => ({
count: counter.count,
name: counter.name,
upStep: counter.upStep,
downStep: counter.downStep,
})); this.dataPreferences?.putSync(Index.counterStorageKey, JSON.stringify(saveData));
this.dataPreferences?.flush();
}

4、用户界面

用户界面的设计采用了现代简洁的风格,主要由顶部的总计数显示区、中间的计数器列表区和底部的操作按钮组成。列表项支持左右滑动以显示重置和删除按钮。

@Builder
itemStart(index: number) {
Row() {
Text('重置').fontColor(Color.White).fontSize('40lpx').textAlign(TextAlign.Center).width('180lpx');
}
.height('100%')
.backgroundColor(Color.Orange)
.justifyContent(FlexAlign.SpaceEvenly)
.borderRadius({ topLeft: 10, bottomLeft: 10 })
.onClick(() => {
this.counters[index].count = 0;
this.updateTotalCount();
this.listScroller.closeAllSwipeActions();
});
}

5、动画效果

当用户添加新的计数器时,通过动画效果让新计数器逐渐放大至正常尺寸,提升用户体验。

this.counters.unshift(new CounterItem(`新计数项${Index.counterId}`));
this.listScroller.scrollTo({ xOffset: 0, yOffset: 0 });
this.counters[0].scale = { x: 0.8, y: 0.8 }; animateTo({
duration: 1000,
curve: curves.springCurve(0, 10, 80, 10),
iterations: 1,
onFinish: () => {}
}, () => {
this.counters[0].scale = { x: 1, y: 1 };
});

【总结】

通过上述步骤,我们成功地构建了一个具备基本功能的计数器应用。在这个过程中,我们不仅学习了如何使用鸿蒙NEXT提供的各种API,还掌握了如何结合动画、数据持久化等技术点来优化用户体验。希望本文能为你的鸿蒙开发之旅提供一些帮助和灵感!

【完整代码】

import { curves, promptAction } from '@kit.ArkUI' // 导入动画曲线和提示操作
import { preferences } from '@kit.ArkData' // 导入偏好设置模块 @ObservedV2
// 观察者装饰器,监控状态变化
class CounterItem {
id: number = ++Index.counterId // 计数器ID,自动递增
@Trace name: string // 计数器名称
@Trace count: number = 0 // 计数器当前值,初始为0
@Trace scale: ScaleOptions = { x: 1, y: 1 } // 计数器缩放比例,初始为1
upStep: number = 1 // 增加步长,初始为1
downStep: number = 1 // 减少步长,初始为1 constructor(name: string) { // 构造函数,初始化计数器名称
this.name = name
}
} @Entry
// 入口组件装饰器
@Component
// 组件装饰器
struct Index {
static counterStorageKey: string = "counterStorageKey" // 存储计数器数据的键
static counterId: number = 0 // 静态计数器ID
@State listSpacing: number = 20 // 列表项间距
@State listItemHeight: number = 120 // 列表项高度
@State baseFontSize: number = 60 // 基础字体大小
@State @Watch('updateTotalCount') counters: CounterItem[] = [] // 计数器数组,监控总计数更新
@State totalCount: number = 0 // 总计数
@State isSheetVisible: boolean = false // 控制底部弹出表单的可见性
@State selectedIndex: number = 0 // 当前选中的计数器索引
listScroller: ListScroller = new ListScroller() // 列表滚动器实例
dataPreferences: preferences.Preferences | undefined = undefined // 偏好设置实例 updateTotalCount() { // 更新总计数的方法
let total = 0; // 初始化总计数
for (let i = 0; i < this.counters.length; i++) { // 遍历计数器数组
total += this.counters[i].count // 累加每个计数器的count值
}
this.totalCount = total // 更新总计数
this.saveDataToLocal() // 保存数据到本地
} saveDataToLocal() { // 保存计数器数据到本地的方法
const saveData: object[] = [] // 初始化保存数据的数组
for (let i = 0; i < this.counters.length; i++) { // 遍历计数器数组
let counter: CounterItem = this.counters[i] // 获取当前计数器
saveData.push(Object({
// 将计数器数据添加到保存数组
count: counter.count,
name: counter.name,
upStep: counter.upStep,
downStep: counter.downStep,
}))
}
this.dataPreferences?.putSync(Index.counterStorageKey, JSON.stringify(saveData)) // 将数据保存到偏好设置
this.dataPreferences?.flush() // 刷新偏好设置
} @Builder
// 构建器装饰器
itemStart(index: number) { // 列表项左侧的重置按钮
Row() {
Text('重置').fontColor(Color.White).fontSize('40lpx')// 显示“重置”文本
.textAlign(TextAlign.Center)// 文本居中
.width('180lpx') // 设置宽度
}
.height('100%') // 设置高度
.backgroundColor(Color.Orange) // 设置背景颜色
.justifyContent(FlexAlign.SpaceEvenly) // 设置内容均匀分布
.borderRadius({ topLeft: 10, bottomLeft: 10 }) // 设置圆角
.onClick(() => { // 点击事件
this.counters[index].count = 0 // 重置计数器的count为0
this.updateTotalCount() // 更新总计数
this.listScroller.closeAllSwipeActions() // 关闭所有滑动操作
})
} @Builder
// 构建器装饰器
itemEnd(index: number) { // 列表项右侧的删除按钮
Row() {
Text('删除').fontColor(Color.White).fontSize('40lpx')// 显示“删除”文本
.textAlign(TextAlign.Center)// 文本居中
.width('180lpx') // 设置宽度
}
.height('100%') // 设置高度
.backgroundColor(Color.Red) // 设置背景颜色
.justifyContent(FlexAlign.SpaceEvenly) // 设置内容均匀分布
.borderRadius({ topRight: 10, bottomRight: 10 }) // 设置圆角
.onClick(() => { // 点击事件
this.counters.splice(index, 1) // 从数组中删除计数器
this.listScroller.closeAllSwipeActions() // 关闭所有滑动操作
promptAction.showToast({
// 显示删除成功的提示
message: '删除成功',
duration: 2000,
bottom: '400lpx'
});
})
} aboutToAppear(): void { // 组件即将出现时调用
const options: preferences.Options = { name: Index.counterStorageKey }; // 获取偏好设置选项
this.dataPreferences = preferences.getPreferencesSync(getContext(), options); // 同步获取偏好设置
const savedData: string = this.dataPreferences.getSync(Index.counterStorageKey, "[]") as string // 获取保存的数据
const parsedData: Array<CounterItem> = JSON.parse(savedData) as Array<CounterItem> // 解析数据
console.info(`parsedData:${JSON.stringify(parsedData)}`) // 打印解析后的数据
for (const item of parsedData) { // 遍历解析后的数据
const newItem = new CounterItem(item.name) // 创建新的计数器实例
newItem.count = item.count // 设置计数器的count
newItem.upStep = item.upStep // 设置计数器的upStep
newItem.downStep = item.downStep // 设置计数器的downStep
this.counters.push(newItem) // 将新计数器添加到数组
}
this.updateTotalCount() // 更新总计数
} build() { // 构建组件的UI
Column() {
Text('计数器')// 显示标题
.width('100%')// 设置宽度
.height('88lpx')// 设置高度
.fontSize('38lpx')// 设置字体大小
.backgroundColor(Color.White)// 设置背景颜色
.textAlign(TextAlign.Center) // 文本居中
Column() {
List({ space: this.listSpacing, scroller: this.listScroller }) { // 创建列表
ForEach(this.counters, (counter: CounterItem, index: number) => { // 遍历计数器数组
ListItem() { // 列表项
Row() { // 行布局
Stack() { // 堆叠布局
Rect().fill("#65DACC").width(`${this.baseFontSize / 2}lpx`).height('4lpx') // 上方横条
Circle()// 圆形按钮
.width(`${this.baseFontSize}lpx`)
.height(`${this.baseFontSize}lpx`)
.fillOpacity(0)// 透明填充
.borderWidth('4lpx')// 边框宽度
.borderRadius('50%')// 圆角
.borderColor("#65DACC") // 边框颜色
}
.width(`${this.baseFontSize * 2}lpx`) // 设置宽度
.height(`100%`) // 设置高度
.clickEffect({ scale: 0.6, level: ClickEffectLevel.LIGHT }) // 点击效果
.onClick(() => { // 点击事件
counter.count -= counter.downStep // 减少计数器的count
this.updateTotalCount() // 更新总计数
}) Stack() { // 堆叠布局
Text(counter.name)// 显示计数器名称
.fontSize(`${this.baseFontSize / 2}lpx`)// 设置字体大小
.fontColor(Color.Gray)// 设置字体颜色
.margin({ bottom: `${this.baseFontSize * 2}lpx` }) // 设置底部边距
Text(`${counter.count}`)// 显示计数器当前值
.fontColor(Color.Black)// 设置字体颜色
.fontSize(`${this.baseFontSize}lpx`) // 设置字体大小
}.height('100%') // 设置高度
Stack() { // 堆叠布局
Rect().fill("#65DACC").width(`${this.baseFontSize / 2}lpx`).height('4lpx') // 下方横条
Rect()
.fill("#65DACC")
.width(`${this.baseFontSize / 2}lpx`)
.height('4lpx')
.rotate({ angle: 90 }) // 垂直横条
Circle()// 圆形按钮
.width(`${this.baseFontSize}lpx`)// 设置宽度
.height(`${this.baseFontSize}lpx`)// 设置高度
.fillOpacity(0)// 透明填充
.borderWidth('4lpx')// 边框宽度
.borderRadius('50%')// 圆角
.borderColor("#65DACC") // 边框颜色
}
.width(`${this.baseFontSize * 2}lpx`) // 设置堆叠布局宽度
.height(`100%`) // 设置堆叠布局高度
.clickEffect({ scale: 0.6, level: ClickEffectLevel.LIGHT }) // 点击效果
.onClick(() => { // 点击事件
counter.count += counter.upStep // 增加计数器的count
this.updateTotalCount() // 更新总计数
})
}
.width('100%') // 设置列表项宽度
.backgroundColor(Color.White) // 设置背景颜色
.justifyContent(FlexAlign.SpaceBetween) // 设置内容两端对齐
.padding({ left: '30lpx', right: '30lpx' }) // 设置左右内边距
}
.height(this.listItemHeight) // 设置列表项高度
.width('100%') // 设置列表项宽度
.margin({
// 设置列表项的外边距
top: index == 0 ? this.listSpacing : 0, // 如果是第一个项,设置上边距
bottom: index == this.counters.length - 1 ? this.listSpacing : 0 // 如果是最后一个项,设置下边距
})
.borderRadius(10) // 设置圆角
.clip(true) // 裁剪超出部分
.swipeAction({ start: this.itemStart(index), end: this.itemEnd(index) }) // 设置滑动操作
.scale(counter.scale) // 设置计数器缩放比例
.onClick(() => { // 点击事件
this.selectedIndex = index // 设置当前选中的计数器索引
this.isSheetVisible = true // 显示底部弹出表单
}) }, (counter: CounterItem) => counter.id.toString())// 使用计数器ID作为唯一键
.onMove((from: number, to: number) => { // 列表项移动事件
const tmp = this.counters.splice(from, 1); // 从原位置移除计数器
this.counters.splice(to, 0, tmp[0]) // 插入到新位置
}) }
.scrollBar(BarState.Off) // 隐藏滚动条
.width('648lpx') // 设置列表宽度
.height('100%') // 设置列表高度
}
.width('100%') // 设置列宽度
.layoutWeight(1) // 设置布局权重 Row() { // 底部合计行
Column() { // 列布局
Text('合计').fontSize('26lpx').fontColor(Color.Gray) // 显示“合计”文本
Text(`${this.totalCount}`).fontSize('38lpx').fontColor(Color.Black) // 显示总计数
}.margin({ left: '50lpx' }) // 设置左边距
.justifyContent(FlexAlign.Start) // 设置内容左对齐
.alignItems(HorizontalAlign.Start) // 设置项目左对齐
.width('300lpx') // 设置列宽度 Row() { // 添加按钮行
Text('添加').fontColor(Color.White).fontSize('28lpx') // 显示“添加”文本
}
.onClick(() => { // 点击事件
this.counters.unshift(new CounterItem(`新计数项${Index.counterId}`)) // 添加新计数器
this.listScroller.scrollTo({ xOffset: 0, yOffset: 0 }) // 滚动到顶部
this.counters[0].scale = { x: 0.8, y: 0.8 }; // 设置新计数器缩放
animateTo({
// 动画效果
duration: 1000, // 动画持续时间
curve: curves.springCurve(0, 10, 80, 10), // 动画曲线
iterations: 1, // 动画迭代次数
onFinish: () => { // 动画完成后的回调
}
}, () => {
this.counters[0].scale = { x: 1, y: 1 }; // 恢复缩放
})
}) .width('316lpx') // 设置按钮宽度
.height('88lpx') // 设置按钮高度
.backgroundColor("#65DACC") // 设置按钮背景颜色
.borderRadius(10) // 设置按钮圆角
.justifyContent(FlexAlign.Center) // 设置内容居中
}.width('100%').height('192lpx').backgroundColor(Color.White) // 设置行宽度和高度 }
.backgroundColor("#f2f2f7") // 设置背景颜色
.width('100%') // 设置宽度
.height('100%') // 设置高度
.bindSheet(this.isSheetVisible, this.mySheet(), {
// 绑定底部弹出表单
height: 300, // 设置表单高度
dragBar: false, // 禁用拖动条
onDisappear: () => { // 表单消失时的回调
this.isSheetVisible = false // 隐藏表单
}
})
} @Builder
// 构建器装饰器
mySheet() { // 创建底部弹出表单
Column({ space: 20 }) { // 列布局,设置间距
Row() { // 行布局
Text('计数标题:') // 显示“计数标题”文本
TextInput({ text: this.counters[this.selectedIndex].name }).width('300lpx').onChange((value) => { // 输入框,绑定计数器名称
this.counters[this.selectedIndex].name = value // 更新计数器名称
}) } Row() { // 行布局
Text('增加步长:') // 显示“增加步长”文本
TextInput({ text: `${this.counters[this.selectedIndex].upStep}` })// 输入框,绑定增加步长
.width('300lpx')// 设置输入框宽度
.type(InputType.Number)// 设置输入框类型为数字
.onChange((value) => { // 输入框变化事件
this.counters[this.selectedIndex].upStep = parseInt(value) // 更新增加步长
this.updateTotalCount() // 更新总计数
}) } Row() { // 行布局
Text('减少步长:') // 显示“减少步长”文本
TextInput({ text: `${this.counters[this.selectedIndex].downStep}` })// 输入框,绑定减少步长
.width('300lpx')// 设置输入框宽度
.type(InputType.Number)// 设置输入框类型为数字
.onChange((value) => { // 输入框变化事件
this.counters[this.selectedIndex].downStep = parseInt(value) // 更新减少步长
this.updateTotalCount() // 更新总计数
}) }
}
.justifyContent(FlexAlign.Start) // 设置内容左对齐
.padding(40) // 设置内边距
.width('100%') // 设置宽度
.height('100%') // 设置高度
.backgroundColor(Color.White) // 设置背景颜色
}
}

  

鸿蒙NEXT开发案例:计数器的更多相关文章

  1. 最全华为鸿蒙 HarmonyOS 开发资料汇总

    开发 本示例基于 OpenHarmony 下的 JavaScript UI 框架,进行项目目录解读,JS FA.常用和自定义组件.用户交互.JS 动画的实现,通过本示例可以基本了解和学习到 JavaS ...

  2. SpringBoot开发案例之多任务并行+线程池处理

    前言 前几篇文章着重介绍了后端服务数据库和多线程并行处理优化,并示例了改造前后的伪代码逻辑.当然了,优化是无止境的,前人栽树后人乘凉.作为我们开发者来说,既然站在了巨人的肩膀上,就要写出更加优化的程序 ...

  3. 使用Jquery+EasyUI 进行框架项目开发案例讲解之五 模块(菜单)管理源码分享

    http://www.cnblogs.com/huyong/p/3454012.html 使用Jquery+EasyUI 进行框架项目开发案例讲解之五  模块(菜单)管理源码分享    在上四篇文章 ...

  4. 使用Jquery+EasyUI 进行框架项目开发案例讲解之四 组织机构管理源码分享

    http://www.cnblogs.com/huyong/p/3404647.html 在上三篇文章  <使用Jquery+EasyUI进行框架项目开发案例讲解之一---员工管理源码分享> ...

  5. 使用Jquery+EasyUI 进行框架项目开发案例讲解之三---角色管理源码分享

    使用Jquery+EasyUI 进行框架项目开发案例讲解之三 角色管理源码分享    在上两篇文章  <使用Jquery+EasyUI进行框架项目开发案例讲解之一---员工管理源码分享> ...

  6. 使用Jquery+EasyUI 进行框架项目开发案例讲解之二---用户管理源码分享

    使用Jquery+EasyUI 进行框架项目开发案例讲解之二 用户管理源码分享   在上一篇文章<使用Jquery+EasyUI进行框架项目开发案例讲解之一---员工管理源码分享>我们分享 ...

  7. 【推荐】使用Jquery+EasyUI进行框架项目开发案例讲解之一---员工管理源码分享

    使用Jquery+EasyUI 进行框架项目开发案例讲解之一 员工管理源码分享   在开始讲解之前,我们先来看一下什么是Jquery EasyUI?jQuery EasyUI是一组基于jQuery的U ...

  8. 使用Jquery+EasyUI进行框架项目开发案例解说之中的一个---员工管理源代码分享

    使用Jquery+EasyUI 进行框架项目开发案例解说之中的一个 员工管理源代码分享 在開始解说之前,我们先来看一下什么是Jquery EasyUI?jQuery EasyUI是一组基于jQuery ...

  9. 百度UEditor开发案例(JSP)

    本案例的开发环境:MyEclipse+tomcat+jdk     本案例的开发内容: 用百度编辑器发布新闻(UEditor的初始化开发部署) 编辑已发过的新闻(UEditor的应用——编辑旧文章) ...

  10. 使用Jquery+EasyUI 进行框架项目开发案例解说之二---用户管理源代码分享

    使用Jquery+EasyUI 进行框架项目开发案例解说之二 用户管理源代码分享  在上一篇文章<使用Jquery+EasyUI进行框架项目开发案例解说之中的一个---员工管理源代码分享> ...

随机推荐

  1. 2024-09-04:用go语言,给定一个长度为n的数组 happiness,表示每个孩子的幸福值,以及一个正整数k,我们需要从这n个孩子中选出k个孩子。 在筛选过程中,每轮选择一个孩子时,所有尚未选

    2024-09-04:用go语言,给定一个长度为n的数组 happiness,表示每个孩子的幸福值,以及一个正整数k,我们需要从这n个孩子中选出k个孩子. 在筛选过程中,每轮选择一个孩子时,所有尚未选 ...

  2. exceptionless 在 windows 上 手动部署,非docker 详细步骤

    关于exceptionless 是什么我就不多说了,能看到这篇文章的都知道了.网上几乎都是docker部署的,docker部署的确十分方便,但是有的人没有条件用docker,像我就不想花这个钱去多服务 ...

  3. iOS 屏幕旋转的实践解析

    ​ 摘要:如何更灵活便捷的实现自定义屏幕旋转场景,本文带你揭秘! 文|即构 iOS 应用开发团队 屏幕旋转是在视频直播类 APP 中常见的场景,在即构科技之前发布的 Roomkit SDK 中也有屏幕 ...

  4. VS code常用插件安装【持续更新】

    Auto Close Tag 自动添加HTML/XML关闭标签.例如,在输入<div>时,输入完最后一个尖括号>时,会自动添加对应的闭合标签</div> Auto Ren ...

  5. CSS & JS Effect – 用 wheel 模拟 scroll

    前言 在 用 JavaScript 实现 position sticky 文章中,我提到了用 wheel 来模拟 scroll 效果. 这篇来说说具体怎么实现,挺简单的哦. Preparation t ...

  6. 从0开始计算机体系结构的学习(一):FGPA预备知识与Vivado环境搭建

    引入与预备知识 什么是FPGA? FPGA(Field-Programmable Gate Array,现场可编程门阵列)是一种集成电路(IC),其硬件功能可以通过用户在现场编程来定义.与传统的ASI ...

  7. Linux命令每天都要使用,但又太长记不住怎么办?教你1个方法

    序言各位好啊,我是会编程的蜗牛,作为java开发者 ,我们肯定会与linux服务器打交道,关于linux服务器的连接工具,可以参考我的文章Tabby,一款老外都在用的 SSH工具,竟然还支持网页操作~ ...

  8. P4036 [JSOI2008] 火星人

    #include <bits/stdc++.h> #define int long long using namespace std; int len; int m; int rt = 0 ...

  9. 课时04:了解HTTP网络协议

    什么是HTTP协议 HTTP(HyperText Transfer Protocol)叫超文本传输协议,它是web服务器和客户端直接进行数据传输的规则,是一个无状态的应用层协议. HTTP协议工作原理 ...

  10. USB type-c CC管脚如何做到正反接检测功能

    USB Type-C 连接器的 CC (Configuration Channel) 管脚用于实现插头方向检测和电源管理.具体来说,USB Type-C 连接器具有两个 CC 管脚:CC1 和 CC2 ...