ladybird/Userland/Applications/Calculator/Keypad.cpp
2021-05-23 23:18:54 +01:00

165 lines
3.8 KiB
C++

/*
* Copyright (c) 2019-2020, Sergey Bugaev <bugaevc@serenityos.org>
* Copyright (c) 2021, Max Wipfli <mail@maxwipfli.ch>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "Keypad.h"
#include <AK/StringBuilder.h>
Keypad::Keypad()
{
}
Keypad::~Keypad()
{
}
void Keypad::type_digit(int digit)
{
u64 previous_value = 0;
switch (m_state) {
case State::External:
m_state = State::TypingInteger;
m_negative = false;
m_int_value = digit;
m_frac_value = 0;
m_frac_length = 0;
break;
case State::TypingInteger:
VERIFY(m_frac_value.value() == 0);
VERIFY(m_frac_length == 0);
previous_value = m_int_value.value();
m_int_value *= 10;
m_int_value += digit;
if (m_int_value.has_overflow())
m_int_value = previous_value;
break;
case State::TypingDecimal:
previous_value = m_frac_value.value();
m_frac_value *= 10;
m_frac_value += digit;
if (m_frac_value.has_overflow())
m_frac_value = previous_value;
else
m_frac_length++;
break;
}
}
void Keypad::type_decimal_point()
{
switch (m_state) {
case State::External:
m_negative = false;
m_int_value = 0;
m_frac_value = 0;
m_frac_length = 0;
m_state = State::TypingDecimal;
break;
case State::TypingInteger:
VERIFY(m_frac_value.value() == 0);
VERIFY(m_frac_length == 0);
m_state = State::TypingDecimal;
break;
case State::TypingDecimal:
// Ignore it.
break;
}
}
void Keypad::type_backspace()
{
switch (m_state) {
case State::External:
m_negative = false;
m_int_value = 0;
m_frac_value = 0;
m_frac_length = 0;
break;
case State::TypingDecimal:
if (m_frac_length > 0) {
m_frac_value /= 10;
m_frac_length--;
break;
}
VERIFY(m_frac_value.value() == 0);
m_state = State::TypingInteger;
[[fallthrough]];
case State::TypingInteger:
VERIFY(m_frac_value.value() == 0);
VERIFY(m_frac_length == 0);
m_int_value /= 10;
if (m_int_value.value() == 0)
m_negative = false;
break;
}
}
double Keypad::value() const
{
double res = 0.0;
u64 frac = m_frac_value.value();
for (int i = 0; i < m_frac_length; i++) {
u8 digit = frac % 10;
res += digit;
res /= 10.0;
frac /= 10;
}
res += m_int_value.value();
if (m_negative)
res = -res;
return res;
}
void Keypad::set_value(double value)
{
m_state = State::External;
if (value < 0.0) {
m_negative = true;
value = -value;
} else
m_negative = false;
m_int_value = value;
value -= m_int_value.value();
m_frac_value = 0;
m_frac_length = 0;
while (value != 0) {
value *= 10.0;
int digit = value;
m_frac_value *= 10;
m_frac_value += digit;
m_frac_length++;
value -= digit;
if (m_frac_length > 6)
break;
}
}
String Keypad::to_string() const
{
StringBuilder builder;
if (m_negative)
builder.append("-");
builder.appendff("{}", m_int_value.value());
// NOTE: This is so the decimal point appears on screen as soon as you type it.
if (m_frac_length > 0 || m_state == State::TypingDecimal)
builder.append('.');
if (m_frac_length > 0) {
// FIXME: This disables the compiletime format string check since we can't parse '}}' here correctly.
// remove the 'StringView { }' when that's fixed.
builder.appendff(StringView { "{:0{}}" }, m_frac_value.value(), m_frac_length);
}
return builder.to_string();
}