mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-26 18:02:06 +00:00 
			
		
		
		
	 8f552c9979
			
		
	
	
		8f552c9979
		
	
	
	
	
		
			
			Calculator now uses the KeypadValue class instead of double in its internal calculations. By not constantly converting to double back-and-forth, we do not use precision simply by, for example, negating a number. This fixes #7484.
		
			
				
	
	
		
			145 lines
		
	
	
	
		
			3.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
	
		
			3.6 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 "KeypadValue.h"
 | |
| #include <AK/Math.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;
 | |
|     }
 | |
| }
 | |
| 
 | |
| KeypadValue Keypad::value() const
 | |
| {
 | |
|     KeypadValue frac_part = { (i64)m_frac_value.value(), m_frac_length };
 | |
|     KeypadValue int_part = { (i64)m_int_value.value() };
 | |
|     KeypadValue res = int_part + frac_part;
 | |
|     if (m_negative)
 | |
|         res = -res;
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| void Keypad::set_value(KeypadValue value)
 | |
| {
 | |
|     m_state = State::External;
 | |
| 
 | |
|     if (value.m_value < 0) {
 | |
|         m_negative = true;
 | |
|         value = -value;
 | |
|     } else
 | |
|         m_negative = false;
 | |
| 
 | |
|     m_int_value = value.m_value / (u64)AK::pow(10.0, (double)value.m_decimal_places);
 | |
|     m_frac_value = value.m_value % (u64)AK::pow(10.0, (double)value.m_decimal_places);
 | |
|     m_frac_length = value.m_decimal_places;
 | |
| }
 | |
| 
 | |
| 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("{:0{}}"sv, m_frac_value.value(), m_frac_length);
 | |
|     }
 | |
| 
 | |
|     return builder.to_string();
 | |
| }
 |