1、前言

经过我们前面10章的学习,我们基本上可以开发出一个简单的APP了,为了巩固学习的内容,我们先开发一个计算器APP来连个手(文末有源代码),界面如下:

  • 包含基本的计算器运算功能

  • 支持一键清空,支持逐个删除数字

  • 支持将上次计算的过程保存下来,下次打开app继续;

  • 支持清理之前保存的记录

这里面我们会用到知识如下:

  1. 开发UI界面(基本控件的使用以及基本布局的使用);

  2. 监听用户点击事件,在输入栏中及时反馈(动态交互);

  3. 支持将过程持久化到本地,下次打开后自动恢复;

let's go!

2、创建项目 & 初始化

打开开发工具 DevEco-Studio,按照以下顺序创建项目,过程不再详述,看截图。

3、UI描述

我们的布局比较简单,分三个大块从上到下,分别是:

  1. 用户输入与计算结果显示部分;

  2. 核心数字键盘部分;

  3. 一个大的=按钮

示意图如下:

我们可以注意到,数字按钮的样式是一样的(蓝色),操作符按钮的样式也是一样的(绿色),三个功能按钮的样式也是一样的(红色),因此,我们可以使用鸿蒙的样式复用只是来减少重复代码量。

我们三部分复用的样式代码如下:

// 数字键盘的公共样式
@Extend(Button) function numberBtn(num: number, click: (num: number) => void) {
.type(ButtonType.Capsule)
.width(60)
.height(60)
.onClick(() => click(num))
}
// 运算符公共样式
@Extend(Button) function operatorBtn() {
.type(ButtonType.Capsule)
.backgroundColor(Color.Green)
.width(60)
.height(60)
} // 额外功能公共样式
@Extend(Button) function extraBtn() {
.type(ButtonType.Capsule)
.width(60)
.height(60)
.backgroundColor(Color.Red)
}

紧接着就是我们的UI描述部分代码,如下:

  build() {
Column() {
Column() {
Text(this.message)
.fontSize(20)
.width('100%')
.textAlign(TextAlign.End)
}
.padding(10)
.backgroundColor('gray')
.borderWidth(1)
.borderRadius(10)
.margin(10) Row() {
Button('7').numberBtn(7, this.onNumberClick)
Button('8').numberBtn(8, this.onNumberClick)
Button('9').numberBtn(9, this.onNumberClick)
Button('重置').extraBtn().onClick(this.onResetClick)
}
.justifyContent(FlexAlign.SpaceAround)
.numberRow() Row() {
Button('4').numberBtn(4, this.onNumberClick)
Button('5').numberBtn(5, this.onNumberClick)
Button('6').numberBtn(6, this.onNumberClick)
Button('删除').extraBtn().onClick(this.onDeleteClick) }
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('1').numberBtn(1, this.onNumberClick)
Button('2').numberBtn(2, this.onNumberClick)
Button('3').numberBtn(3, this.onNumberClick)
Button('清理').extraBtn().onClick(this.onClearClick)
}
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('%')
.type(ButtonType.Capsule)
.width(60)
.height(60)
Button('0').numberBtn(0, this.onNumberClick)
Button('.')
.type(ButtonType.Capsule)
.width(60)
.height(60)
.onClick(this.onDotClick)
Button('保存').operatorBtn().onClick(this.onSaveClick)
}
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('x')
.operatorBtn()
.onClick(() => this.onOperatorClick('x'))
Button('÷')
.operatorBtn()
.onClick(() => this.onOperatorClick('÷'))
Button('+')
.operatorBtn()
.onClick(() => this.onOperatorClick('+'))
Button('-')
.operatorBtn()
.onClick(() => this.onOperatorClick('-'))
}
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('=')
.width('100%')
.backgroundColor(Color.Pink)
.onClick(this.onCalcClick)
}
.margin(10)
}
.height('100%')
}

4、用户交互

紧接着,我们来实现用户的功能交互

  • 用户点击数字按钮时,监听器事件
  onNumberClick = (num: number) => {
if (this.message === '0') {
this.message = `${num}`;
} else {
this.message += `${num}`
}
}
  • 用户点击运算符按钮时的监听器事件
  onOperatorClick = (_opt: string) => {
this.message += _opt;
this.calcFinished = false;
}
  • 数据持久化和清空持久化数据
  onSaveClick = () => {
AppStorage.SetOrCreate('lastResult', this.message);
}
onClearClick = () => {
PersistentStorage.DeleteProp('lastResult')
this.message = '0';
}

其他的功能感兴趣可以在文末看详细的源代码。演示如下(包含了数据持久化,现场恢复的演示)

5、源代码

代码仓库地址:

https://gitee.com/lantingshuxu/harmony-calculator

速览核心的UI代码:

// 数字键盘的公共样式
@Extend(Button) function numberBtn(num: number, click: (num: number) => void) {
.type(ButtonType.Capsule)
.width(60)
.height(60)
.onClick(() => click(num))
}
// 运算符公共样式
@Extend(Button) function operatorBtn() {
.type(ButtonType.Capsule)
.backgroundColor(Color.Green)
.width(60)
.height(60)
}
// 额外功能公共样式
@Extend(Button) function extraBtn() {
.type(ButtonType.Capsule)
.width(60)
.height(60)
.backgroundColor(Color.Red)
} PersistentStorage.PersistProp('lastResult', '0'); type Operator = '+' | '-' | '*' | '/' | '(' | ')';
// 传入字符串,运算其结果(可以跳过)
function calculate(expression: string): number {
    //.. 这部分代码是计算用户输入结果d的,不重要,省略
} @Entry
@Component
struct Index {
@StorageProp('lastResult') message: string = '0'
calcFinished: boolean = true; @Styles numberRow() {
.width('100%')
.margin({ bottom: 10 })
} build() {
Column() {
Column() {
Text(this.message)
.fontSize(20)
.width('100%')
.textAlign(TextAlign.End)
}
.padding(10)
.backgroundColor('gray')
.borderWidth(1)
.borderRadius(10)
.margin(10) Row() {
Button('7').numberBtn(7, this.onNumberClick)
Button('8').numberBtn(8, this.onNumberClick)
Button('9').numberBtn(9, this.onNumberClick)
Button('重置').extraBtn().onClick(this.onResetClick)
}
.justifyContent(FlexAlign.SpaceAround)
.numberRow() Row() {
Button('4').numberBtn(4, this.onNumberClick)
Button('5').numberBtn(5, this.onNumberClick)
Button('6').numberBtn(6, this.onNumberClick)
Button('删除').extraBtn().onClick(this.onDeleteClick) }
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('1').numberBtn(1, this.onNumberClick)
Button('2').numberBtn(2, this.onNumberClick)
Button('3').numberBtn(3, this.onNumberClick)
Button('清理').extraBtn().onClick(this.onClearClick)
}
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('%')
.type(ButtonType.Capsule)
.width(60)
.height(60)
Button('0').numberBtn(0, this.onNumberClick)
Button('.')
.type(ButtonType.Capsule)
.width(60)
.height(60)
.onClick(this.onDotClick)
Button('保存').operatorBtn().onClick(this.onSaveClick)
}
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('x')
.operatorBtn()
.onClick(() => this.onOperatorClick('x'))
Button('÷')
.operatorBtn()
.onClick(() => this.onOperatorClick('÷'))
Button('+')
.operatorBtn()
.onClick(() => this.onOperatorClick('+'))
Button('-')
.operatorBtn()
.onClick(() => this.onOperatorClick('-'))
}
.numberRow()
.justifyContent(FlexAlign.SpaceAround) Row() {
Button('=')
.width('100%')
.backgroundColor(Color.Pink)
.onClick(this.onCalcClick)
}
.margin(10)
}
.height('100%')
} resetCheck() {
if (this.calcFinished) {
this.message = '0';
this.calcFinished = false;
}
} onNumberClick = (num: number) => {
this.resetCheck();
if (this.message === '0') {
this.message = `${num}`;
} else {
this.message += `${num}`
}
}
onOperatorClick = (_opt: string) => {
this.message += _opt;
this.calcFinished = false;
}
onDotClick = () => {
this.resetCheck();
this.message += '.';
}
onResetClick = () => {
this.message = '0';
this.calcFinished = true;
}
onDeleteClick = () => {
this.resetCheck();
if (this.message.length > 1) {
this.message = this.message.substr(0, this.message.length - 1);
} else {
this.message = '0'
}
}
onSaveClick = () => {
AppStorage.SetOrCreate('lastResult', this.message);
}
onClearClick = () => {
PersistentStorage.DeleteProp('lastResult')
this.message = '0';
}
onCalcClick = () => {
this.message = calculate(this.message.replace(/x/g, '*').replace(/÷/g, '/')) + '';
this.calcFinished = true;
}
}

6、结语

从代码我们可以看到,依旧存在很多相似的重复代码,我们还可以优化下吗?

答案是可以的,使用条件渲染。

请持续关注 "鸿蒙UI开发快速入门 —— part11"

鸿蒙UI开发快速入门 —— part11: 鸿蒙计算器开发实践的更多相关文章

  1. HealthKit开发快速入门教程之HealthKit开发概述简介

    HealthKit开发快速入门教程之HealthKit开发概述简介 2014年6月2日召开的年度开发者大会上,苹果发布了一款新的移动应用平台,可以收集和分析用户的健康数据.该移动应用平台被命名为“He ...

  2. Transform组件C#游戏开发快速入门

    Transform组件C#游戏开发快速入门大学霸 组件(Component)可以看作是一类属性的总称.而属性是指游戏对象上一切可设置.调节的选项,如图2-8所示.本文选自C#游戏开发快速入门大学霸   ...

  3. HealthKit开发快速入门教程之HealthKit数据的操作

    HealthKit开发快速入门教程之HealthKit数据的操作 数据的表示 在HealthKit中,数据是最核心的元素.通过分析数据,人们可以看到相关的健康信息.例如,通过统计步数数据,人们可以知道 ...

  4. HealthKit开发快速入门教程之HealthKit框架体系创建健康AppID

    HealthKit开发快速入门教程之HealthKit框架体系创建健康AppID HealthKit开发准备工作 在开发一款HealthKit应用程序时,首先需要讲解HealthKit中有哪些类,在i ...

  5. Apple Watch开发快速入门教程

     Apple Watch开发快速入门教程  试读下载地址:http://pan.baidu.com/s/1eQ8JdR0 介绍:苹果为Watch提供全新的开发框架WatchKit.本教程是国内第一本A ...

  6. 游戏控制杆OUYA游戏开发快速入门教程

    游戏控制杆OUYA游戏开发快速入门教程 1.2.2  游戏控制杆 游戏控制杆各个角度的视图,如图1-4所示,它的硬件规格是本文选自OUYA游戏开发快速入门教程大学霸: 图1-4  游戏控制杆各个角度的 ...

  7. webpack快速入门——实战技巧:开发和生产并行设置

    package.json中,devDependencies和dependencies是不同的 devDependencies:开发依赖 dependencies:生产依赖(线上) 1.安装生产环境的依 ...

  8. SpringBoot开发快速入门

    SpringBoot开发快速入门 目录 一.Spring Boot 入门 1.Spring Boot 简介 2.微服务 3.环境准备 1.maven设置: 2.IDEA设置 4.Spring Boot ...

  9. WPF开发快速入门【7】WPF的拖放功能(Drag and Drop)

    概述 本文描述WPF的拖放功能(Drag and Drop). 拖放功能涉及到两个功能,一个就是拖,一个是放.拖放可以发生在两个控件之间,也可以在一个控件自己内部拖放.假设界面上有两个控件,一个Tre ...

  10. Android开发快速入门(环境配置、Android Studio安装)

    Android是一种激动人心的开源移动平台,它像手机一样无处不在,得到了Google以及其他一些开放手机联盟成员(如三星.HTC.中国移动.Verizon和AT&T等)的支持,因而不能不加以学 ...

随机推荐

  1. 递归获取zip中的文件

    1 //tempPath 系统临时文件夹 2 2 private List<String> getWsContentList(String tempPath) { 3 3 //wsFile ...

  2. scala安装及配置

    window 上安装 Scala 1.Scala下载网址:https://www.scala-lang.org/download/ 2.下载后,双击 msi 文件,一步步安装即可,安装过程你可以使用默 ...

  3. .Net 中带有 ? 的运算符

    // 带 ? 的表达式 // 1. 三元表达式 // 2. ?? 双问号 // obj1 ?? obj2 如果 obj1 为 空(null) 返回 obj2 // Configure the HTTP ...

  4. 新建数据库 phpStudy

    官网:https://www.xp.cn/下载phpStudy : 环境配置:1. 下载MySQL8.0.12 2. 安装HeidiSQL11.0 开启 MySQL8.0.12 修改MySQL的密码: ...

  5. 14. Vue2 和 Vue3 区别

    主要分为四点: 1. Vue3 使用了 proxy 替代了 Object.defineProperty 实现响应式数据 ,所以 vue3 的性能得到了提升 : 2. Vue3 可以在 template ...

  6. druid连接池报错:sql injection violation, multi-statement not allow

    druid连接池报错:sql injection violation, multi-statement not allow 需要配置druid的 multi-statement-allow属性为tru ...

  7. Oracle ASM 常用巡检脚本

    1.查看磁盘组 sqlplus "/ as sysasm" set line 200 set pagesize 200 select group_number,name,state ...

  8. 快速搭建k8s

    换桥接模式,换sealos 桥接模式 部署出问题了,用这个: ipconfig 以太网适配器 以太网: 连接特定的 DNS 后缀 . . . . . . . : 本地链接 IPv6 地址. . . . ...

  9. vue中事件总线bus的用法

    ./util/Bus.js import Bus from 'vue'; let install = function (Vue) { // 设置eventBus Vue.prototype.bus ...

  10. 顺序表(python)

    文章目录 1.创建顺序表 2.按址查找元素的位置 3.增加元素 3.1在头部增加元素 3.2在尾部增加元素 3.3在中间任意位置增加元素 4.删除元素 4.1删除第一个元素 4.2删除指定的元素 5. ...