Files
create/src/Base/UnitsApi.cpp
2011-10-10 13:44:52 +00:00

267 lines
7.8 KiB
C++

/***************************************************************************
* Copyright (c) 2009 Juergen Riegel (FreeCAD@juergen-riegel.net) *
* *
* This file is part of the FreeCAD CAx development system. *
* *
* This library is free software; you can redistribute it and/or *
* modify it under the terms of the GNU Library General Public *
* License as published by the Free Software Foundation; either *
* version 2 of the License, or (at your option) any later version. *
* *
* This library 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 Library General Public License for more details. *
* *
* You should have received a copy of the GNU Library General Public *
* License along with this library; see the file COPYING.LIB. If not, *
* write to the Free Software Foundation, Inc., 59 Temple Place, *
* Suite 330, Boston, MA 02111-1307, USA *
* *
***************************************************************************/
#include "PreCompiled.h"
#ifdef __GNUC__
# include <unistd.h>
#endif
#include <QString>
#include "Exception.h"
#include "UnitsApi.h"
#include "UnitsSchemaInternal.h"
#include "UnitsSchemaImperial1.h"
#include "UnitsSchemaMKS.h"
//#include "UnitsApiPy.h"
#ifndef M_PI
#define M_PI 3.14159265358979323846
#endif
#ifndef M_E
#define M_E 2.71828182845904523536
#endif
#ifndef DOUBLE_MAX
# define DOUBLE_MAX 1.7976931348623157E+308 /* max decimal value of a "double"*/
#endif
#ifndef DOUBLE_MIN
# define DOUBLE_MIN 2.2250738585072014E-308 /* min decimal value of a "double"*/
#endif
using namespace Base;
// suppress annoying warnings from generated source files
#ifdef _MSC_VER
# pragma warning(disable : 4003)
# pragma warning(disable : 4018)
# pragma warning(disable : 4065)
# pragma warning( disable : 4273 )
# pragma warning(disable : 4335) // disable MAC file format warning on VC
#endif
// === names =============================================================
char *QuantityNames[] = {
"length" ,
"area" ,
"volume" ,
"angle" ,
"time span" ,
"velocity" ,
"acceleration",
"mass" ,
"temperature"
};
const QString UnitsApi::getQuantityName(QuantityType t)
{
// check limits
assert(t<9);
// returns
return QString::fromLatin1(QuantityNames[t]);
}
// === static attributes ================================================
UnitsSchema *UnitsApi::UserPrefSystem = new UnitsSchemaInternal();
double UnitsApi::UserPrefFactor [50];
QString UnitsApi::UserPrefUnit [50];
UnitsApi::UnitsApi(const char* filter)
{
bool temp;
Result = parse(filter,temp);
}
UnitsApi::UnitsApi(const std::string& filter)
{
bool temp;
Result = parse(filter.c_str(),temp);
}
UnitsApi::~UnitsApi()
{
}
void UnitsApi::setSchema(UnitSystem s)
{
delete UserPrefSystem;
switch (s) {
case SI1 : UserPrefSystem = new UnitsSchemaInternal(); break;
case SI2 : UserPrefSystem = new UnitsSchemaMKS(); break;
case Imperial1: UserPrefSystem = new UnitsSchemaImperial1(); break;
}
UserPrefSystem->setSchemaUnits();
}
double UnitsApi::translateUnit(const char* str)
{
bool temp;
return parse(str,temp );
}
double UnitsApi::translateUnit(const QString & str)
{
bool temp;
return parse(str.toUtf8() ,temp);
}
// === static translation methodes ==========================================
QString UnitsApi::toStrWithUserPrefs(QuantityType t,double Value)
{
return UserPrefSystem->toStrWithUserPrefs(t,Value);
//double UnitValue = Value/UserPrefFactor[t];
//return QString::fromAscii("%1 %2").arg(UnitValue).arg(UserPrefUnit[t]);
}
void UnitsApi::toStrWithUserPrefs(QuantityType t,double Value,QString &outValue,QString &outUnit)
{
UserPrefSystem->toStrWithUserPrefs(t,Value,outValue,outUnit);
}
PyObject *UnitsApi::toPyWithUserPrefs(QuantityType t,double Value)
{
return PyFloat_FromDouble(Value * UserPrefFactor[t]);
}
double UnitsApi::toDblWithUserPrefs(QuantityType t,const QString & Str)
{
return toDblWithUserPrefs(t,(const char*) Str.toUtf8());
}
double UnitsApi::toDblWithUserPrefs(QuantityType t,const char* Str)
{
bool UsedUnit;
double Value = parse( Str,UsedUnit );
if (UsedUnit)
return Value;
else
return toDblWithUserPrefs(t,Value);
}
double UnitsApi::toDblWithUserPrefs(QuantityType t,double UserVal)
{
return UserVal*UserPrefFactor[t];
}
double UnitsApi::toDblWithUserPrefs(QuantityType t,PyObject *ArgObj)
{
if (PyString_Check(ArgObj))
return toDblWithUserPrefs(t,PyString_AsString(ArgObj));
else if (PyFloat_Check(ArgObj))
return toDblWithUserPrefs(t,PyFloat_AsDouble(ArgObj));
else if (PyInt_Check(ArgObj))
return toDblWithUserPrefs(t,(double)PyInt_AsLong(ArgObj));
else
throw Base::Exception("Wrong parameter type!");
}
void UnitsApi::setPrefOf(QuantityType t,const char* Str)
{
double Factor = translateUnit(Str);
UserPrefUnit[t] = QString::fromLatin1(Str);
UserPrefFactor[t] = Factor;
}
const QString & UnitsApi::getPrefUnitOf(QuantityType t)
{
return UserPrefUnit[t];
}
const double UnitsApi::getPrefFactorOf(QuantityType t)
{
return UserPrefFactor[t];
}
void UnitsApi::setDefaults(void)
{
setPrefOf( Length ,"mm" );
setPrefOf( Area ,"mm^2" );
setPrefOf( Volume ,"mm^3" );
setPrefOf( Angle ,"deg" );
setPrefOf( TimeSpan ,"s" );
setPrefOf( Velocity ,"mm/s" );
setPrefOf( Acceleration ,"mm/s^2" );
setPrefOf( Mass ,"kg" );
setPrefOf( Temperature ,"K" );
}
// === Parser & Scanner stuff ===============================================
// include the Scanner and the Parser for the filter language
double ScanResult=0;
bool UU = false;
// error func
void Unit_yyerror(char *errorinfo)
{ throw Base::Exception(errorinfo); }
// for VC9 (isatty and fileno not supported anymore)
#ifdef _MSC_VER
int isatty (int i) {return _isatty(i);}
int fileno(FILE *stream) {return _fileno(stream);}
#endif
namespace UnitParser {
// show the parser the lexer method
#define yylex UnitsApilex
int UnitsApilex(void);
// Parser, defined in UnitsApi.y
#include "UnitsApi.tab.c"
#ifndef DOXYGEN_SHOULD_SKIP_THIS
// Scanner, defined in UnitsApi.l
#include "lex.UnitsApi.c"
#endif // DOXYGEN_SHOULD_SKIP_THIS
}
double UnitsApi::parse(const char* buffer,bool &UsedUnit)
{
// parse from buffer
UnitParser::YY_BUFFER_STATE my_string_buffer = UnitParser::UnitsApi_scan_string (buffer);
// set the global return variables
ScanResult = DOUBLE_MIN;
UU = false;
// run the parser
UnitParser::Unit_yyparse ();
UsedUnit = UU;
UU=false;
// free the scan buffer
UnitParser::UnitsApi_delete_buffer (my_string_buffer);
if (ScanResult == DOUBLE_MIN)
throw Base::Exception("Unknown error in Unit expression");
return ScanResult;
}