使用 SwiftUI 来做仿做一个原生计算器(技术篇)
概述
这里我们将会使用MVVM架构来编写这个项目,本文将主要探讨计算器计算功能实现的步骤。
代码已开源:https://github.com/Jo-CRuiSe/SwiftUICalculator
Views
calcButtonView
:单个按钮样式
contentsView
:计算器app样式
ViewModel(ContentViewModel
)
变量定义
1
2
3
4
5
6
7
8
9
10
11
12
13
@Published var display: String = "0" //计算器显示字符
@Published var action: Action? //按键操作
@Published var constantlyLit: Bool = false //操作是否常亮
@Published var ACPressed = false //是否按下AC键
private var operandStack: [String] = [] //数字栈
private var operatorStack: [String] = [] //符号栈
private var lastPressedButton: lastOprationType = .clear//上次按下按钮类型
private var shouldClearAll: Bool = true //清除全部栈还是清除当前数字栈的最后一位
private var lastNum = "" //上次按下数字
private var lastOperator = "" //上次按下操作
var canDeleteLastDigit = true //是否允许左右滑动显示界面来删除最后一位
var stacked = false //是否有元素入栈
基本运算逻辑
按键类型
- 数字(0~9)
- 运算(+,-,×,÷ )
- 符号切换键(+/-)
- 百分号(%)
- 等于(=)
- 清除(AC)
数字处理逻辑
追加:当上一个输入的是数字时,追加新数字。如果数字栈中最后一个元素是0,则直接显示当前数字。数字除去小数点后最大长度为9。
清除重新显示:当上次进行了运算,显示当前数字;若上次进行了特殊操作或清除,先清空栈再显示当前数字
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
if lastPressedButton == .operand {
var operand = operandStack.popLast() ?? "0"
if operand == "0" {
operand = text
} else {
operand.append(text)
}
operandStack.append(operand)
lastPressedButton = .operand
} else if lastPressedButton == .clear || lastPressedButton == .specialOperator {
_ = operandStack.popLast()
operandStack.append(text)
}
else {
display = text
operandStack.append(text)
}
lastPressedButton = .operand
运算处理逻辑
- 更改算符:上一个输入也为算符,并且当前操作不是 = 或 +/- 则替换符号栈最后一个符号
1
2
3
if lastPressedButton == .operation && action != .equals && action != .plusMinus{
_ = operatorStack.popLast()
}
- 压入运算符栈:上一个运算符优先级小于等于当前运算符优先级
- + - × ÷进行运算:上一个运算符优先级大于当前运算符优先级
当输入运算符后,运算符栈内只可能有3种情况:
- 有一个运算符:运算符为刚输入的运算符,则不做任何操作
- 有两个运算符:可能组合为低优先级+高优先级,或高优先级+低优先级,后者对高优先级进行处理
- 有三个运算符:可能组合为低优先级+高优先级+高优先级,例如+ × ×,或低优先级+高优先级+低优先级,例如+ × +
符号切换键处理逻辑
- 当上个操作为运算符时,将“-0”压入数字栈
- 当上个操作不为运算符时,取出上个数字,并取相反数
1
2
3
4
5
6
7
8
9
10
11
12
13
14
if lastPressedButton == .operation {
operandStack.append("-0")
display = "-0"
} else {
var last = operandStack.popLast() ?? "0"
if last.first == "-" {
last.removeFirst()
} else {
last = "-" + last
}
operandStack.append(last)
display = last
}
lastPressedButton = .plusMinus
百分号处理逻辑
- 数字栈中最后一个元素 ÷ 100
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
canDeleteLastDigit = false
if operandStack.last == "0" && operandStack.last == nil {
lastPressedButton = .percent
return
} else if lastPressedButton == .operand || lastPressedButton == .percent || lastPressedButton == .equal {
let numString = operandStack.popLast() ?? "0"
let num = (convertToDouble(numString) ?? 0.0) / 100.0
operandStack.append(String(num))
display = convertToString(num)
} else if lastPressedButton == .operation {
let numString = operandStack.popLast() ?? "0"
let num = (convertToDouble(numString) ?? 0.0) / 100.0
operandStack.append(numString)
operandStack.append(String(num))
display = convertToString(num)
}
lastPressedButton = .percent
等于号处理逻辑
将数字栈与符号栈进行组合运算
按下等于号前,符号栈内可能情况为
- 无运算符:则重复运算上次运算符与操作数组合。5-2*3 -> -1 -3 -9 -27
- 有一个运算符:
- 有两个运算符:组合为低优先级+高优先级
- 有三个运算符
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
private func equalButtonPressed() {
canDeleteLastDigit = false
if lastPressedButton == .operation,
let operand = operandStack.last
{
operandStack.append(operand)
}
if operatorStack.count == 1{
guard
let operation = operatorStack.popLast(),
let rightNum = operandStack.popLast(),
let leftNum = operandStack.popLast()
else {return}
lastOperator = operation
lastNum = rightNum
let answer = solve(leftNum: leftNum, operation: operation, rightNum: rightNum)
operandStack.append(answer)
display = answer
} else if operatorStack.count == 2{
guard
let secondOperator = operatorStack.popLast(),
let firstOperator = operatorStack.popLast(),
let thirdNum = operandStack.popLast(),
let secondNum = operandStack.popLast(),
let firstNum = operandStack.popLast()
else { return }
lastOperator = secondOperator
lastNum = thirdNum
let answer1 = solve(leftNum: secondNum, operation: secondOperator, rightNum: thirdNum)
let answer2 = solve(leftNum: firstNum, operation: firstOperator, rightNum: answer1)
operandStack.append(answer2)
display = answer2
} else {
// count == 0
if lastNum == "" && lastOperator == ""
{
display = operandStack.last ?? "0"
} else {
let operand = operandStack.popLast() ?? "0"
let answer = solve(leftNum: operand, operation: lastOperator, rightNum: lastNum)
operandStack.append(answer)
display = answer
}
}
}
清除处理逻辑
恢复初始设置
1
2
3
4
5
6
7
8
9
10
11
12
private func clearAll() {
display = "0"
action = nil
constantlyLit = false
canDeleteLastDigit = false
ACPressed = true
operandStack = []
operatorStack = []
stacked = false
lastNum = ""
lastOperator = ""
}
显示处理逻辑
- 数字栈为空则显示0
- 数字栈不为空则显示栈内最后一个元素
- 显示最大长度为9个字符,长于9个使用科学计数法,2.81531e10,3.362826e9,除去小数点和负号,最长显示9个字符
- 对超过显示范围的数字使用科学计数法(代码中限制了精度为10^{-15})
2.457343726 -> 2.45734373
2.81531e10
3.72529e
-9
测试集
3 + 2 × 5 - 4
3 × 2 + 5 × 4
3 + 2 = % % % % %
1 ÷ 4 = = = = = = = = = = = = = = =
123 × = = = = =
5 + +/- 3 =
5+ +/- =
附加功能
表情雨
本文由作者按照 CC BY 4.0 进行授权