2010-05-17 20:53:04 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2014-02-18 02:34:18 +01:00
|
|
|
*
|
2010-05-17 20:53:04 +00:00
|
|
|
*/
|
|
|
|
|
2010-09-06 14:59:13 +00:00
|
|
|
#include "common/debug.h"
|
2010-05-17 20:53:04 +00:00
|
|
|
#include "common/rational.h"
|
|
|
|
#include "common/util.h"
|
|
|
|
#include "common/algorithm.h"
|
|
|
|
|
|
|
|
namespace Common {
|
|
|
|
|
|
|
|
Rational::Rational() {
|
|
|
|
_num = 1;
|
|
|
|
_denom = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational::Rational(int num) {
|
|
|
|
_num = num;
|
|
|
|
_denom = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational::Rational(int num, int denom) {
|
|
|
|
assert(denom != 0);
|
|
|
|
|
2010-05-17 22:08:40 +00:00
|
|
|
if (denom > 0) {
|
|
|
|
_num = num;
|
|
|
|
_denom = denom;
|
|
|
|
} else {
|
|
|
|
_num = -num;
|
|
|
|
_denom = -denom;
|
|
|
|
}
|
2010-05-17 20:53:04 +00:00
|
|
|
|
2010-05-17 22:08:40 +00:00
|
|
|
cancel();
|
2010-05-17 20:53:04 +00:00
|
|
|
}
|
|
|
|
|
2019-11-10 21:04:29 +01:00
|
|
|
Rational::Rational(const Rational &rational) {
|
|
|
|
_num = rational._num;
|
|
|
|
_denom = rational._denom;
|
|
|
|
}
|
|
|
|
|
2010-05-17 20:53:04 +00:00
|
|
|
void Rational::cancel() {
|
|
|
|
// Cancel the fraction by dividing both the num and the denom
|
2010-05-17 22:07:58 +00:00
|
|
|
// by their greatest common divisor.
|
2010-05-17 20:53:04 +00:00
|
|
|
|
2011-02-09 00:12:02 +00:00
|
|
|
const int gcd = Common::gcd(_num, _denom);
|
2010-05-17 20:53:04 +00:00
|
|
|
|
|
|
|
_num /= gcd;
|
|
|
|
_denom /= gcd;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator=(const Rational &right) {
|
|
|
|
_num = right._num;
|
|
|
|
_denom = right._denom;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator=(int right) {
|
|
|
|
_num = right;
|
|
|
|
_denom = 1;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator+=(const Rational &right) {
|
2011-02-09 00:12:02 +00:00
|
|
|
// Cancel common factors to avoid unnecessary overflow.
|
|
|
|
// Note that the result is *not* always normalized.
|
|
|
|
const int gcd = Common::gcd(_denom, right._denom);
|
|
|
|
|
|
|
|
_num = _num * (right._denom / gcd);
|
|
|
|
_denom = _denom / gcd;
|
|
|
|
_num += right._num * _denom;
|
|
|
|
_denom *= right._denom;
|
2010-05-17 20:53:04 +00:00
|
|
|
|
2010-05-17 22:08:40 +00:00
|
|
|
cancel();
|
2010-05-17 20:53:04 +00:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator-=(const Rational &right) {
|
2011-02-09 00:12:02 +00:00
|
|
|
// Cancel common factors to avoid unnecessary overflow.
|
|
|
|
// Note that the result is *not* always normalized.
|
|
|
|
const int gcd = Common::gcd(_denom, right._denom);
|
|
|
|
|
|
|
|
_num = _num * (right._denom / gcd);
|
|
|
|
_denom = _denom / gcd;
|
|
|
|
_num -= right._num * _denom;
|
|
|
|
_denom *= right._denom;
|
2010-05-17 20:53:04 +00:00
|
|
|
|
2010-05-17 22:08:40 +00:00
|
|
|
cancel();
|
2010-05-17 20:53:04 +00:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator*=(const Rational &right) {
|
2010-05-17 22:08:40 +00:00
|
|
|
// Cross-cancel to avoid unnecessary overflow;
|
|
|
|
// the result then is automatically normalized
|
2011-08-06 09:47:19 +02:00
|
|
|
const int gcd1 = gcd(_num, right._denom);
|
|
|
|
const int gcd2 = gcd(right._num, _denom);
|
2010-05-17 20:53:04 +00:00
|
|
|
|
|
|
|
_num = (_num / gcd1) * (right._num / gcd2);
|
|
|
|
_denom = (_denom / gcd2) * (right._denom / gcd1);
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator/=(const Rational &right) {
|
2010-05-17 22:08:40 +00:00
|
|
|
return *this *= right.getInverse();
|
2010-05-17 20:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator+=(int right) {
|
|
|
|
return *this += Rational(right);
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator-=(int right) {
|
|
|
|
return *this -= Rational(right);
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator*=(int right) {
|
|
|
|
return *this *= Rational(right);
|
|
|
|
}
|
|
|
|
|
|
|
|
Rational &Rational::operator/=(int right) {
|
|
|
|
return *this /= Rational(right);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator-() const {
|
|
|
|
return Rational(-_num, _denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator+(const Rational &right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp += right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator-(const Rational &right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp -= right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator*(const Rational &right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp *= right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator/(const Rational &right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp /= right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator+(int right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp += right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator-(int right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp -= right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator*(int right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp *= right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational Rational::operator/(int right) const {
|
|
|
|
Rational tmp = *this;
|
|
|
|
tmp /= right;
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator==(const Rational &right) const {
|
|
|
|
return (_num == right._num) && (_denom == right._denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator!=(const Rational &right) const {
|
|
|
|
return (_num != right._num) || (_denom != right._denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator>(const Rational &right) const {
|
|
|
|
return (_num * right._denom) > (right._num * _denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator<(const Rational &right) const {
|
|
|
|
return (_num * right._denom) < (right._num * _denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator>=(const Rational &right) const {
|
|
|
|
return (_num * right._denom) >= (right._num * _denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator<=(const Rational &right) const {
|
|
|
|
return (_num * right._denom) <= (right._num * _denom);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator==(int right) const {
|
|
|
|
return (_denom == 1) && (_num == right);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator!=(int right) const {
|
2010-09-04 23:45:23 +00:00
|
|
|
return (_denom != 1) || (_num != right);
|
2010-05-17 20:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator>(int right) const {
|
|
|
|
return *this > Rational(right, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator<(int right) const {
|
|
|
|
return *this < Rational(right, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator>=(int right) const {
|
|
|
|
return *this >= Rational(right, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Rational::operator<=(int right) const {
|
|
|
|
return *this <= Rational(right, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Rational::invert() {
|
|
|
|
assert(_num != 0);
|
|
|
|
|
|
|
|
SWAP(_num, _denom);
|
|
|
|
|
2010-05-17 22:08:40 +00:00
|
|
|
if (_denom < 0) {
|
|
|
|
_denom = -_denom;
|
|
|
|
_num = -_num;
|
|
|
|
}
|
2010-05-17 20:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Rational Rational::getInverse() const {
|
|
|
|
Rational inverse = *this;
|
|
|
|
|
|
|
|
inverse.invert();
|
|
|
|
|
|
|
|
return inverse;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Rational::toInt() const {
|
|
|
|
return _num / _denom;
|
|
|
|
}
|
|
|
|
|
|
|
|
double Rational::toDouble() const {
|
2010-05-17 22:17:42 +00:00
|
|
|
return ((double)_num) / ((double)_denom);
|
2010-05-17 20:53:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
frac_t Rational::toFrac() const {
|
|
|
|
return (_num * FRAC_ONE) / _denom;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational operator+(int left, const Rational &right) {
|
2010-09-11 10:03:31 +00:00
|
|
|
Rational tmp(left);
|
|
|
|
tmp += right;
|
2010-05-17 20:53:04 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational operator-(int left, const Rational &right) {
|
2010-09-11 10:03:31 +00:00
|
|
|
Rational tmp(left);
|
|
|
|
tmp -= right;
|
2010-05-17 20:53:04 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational operator*(int left, const Rational &right) {
|
2010-09-11 10:03:31 +00:00
|
|
|
Rational tmp(left);
|
|
|
|
tmp *= right;
|
2010-05-17 20:53:04 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Rational operator/(int left, const Rational &right) {
|
2010-09-11 10:03:31 +00:00
|
|
|
Rational tmp(left);
|
|
|
|
tmp /= right;
|
2010-05-17 20:53:04 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2010-09-06 14:59:13 +00:00
|
|
|
void Rational::debugPrint(int debuglevel, const char *caption) const {
|
|
|
|
debug(debuglevel, "%s %d/%d", caption, _num, _denom);
|
|
|
|
}
|
|
|
|
|
2010-05-17 20:53:04 +00:00
|
|
|
bool operator==(int left, const Rational &right) {
|
|
|
|
return right == left;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(int left, const Rational &right) {
|
|
|
|
return right != left;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator>(int left, const Rational &right) {
|
|
|
|
return right < left;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator<(int left, const Rational &right) {
|
|
|
|
return right > left;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator>=(int left, const Rational &right) {
|
|
|
|
return right <= left;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator<=(int left, const Rational &right) {
|
|
|
|
return right >= left;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // End of namespace Common
|