kdelibs/kdecore/services/ktraderparsetree.cpp

705 lines
16 KiB
C++
Raw Normal View History

2014-11-13 01:04:59 +02:00
/* This file is part of the KDE project
Copyright (C) 1998, 1999 Torben Weis <weis@kde.org>
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., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "ktraderparsetree_p.h"
namespace KTraderParse {
bool ParseTreeOR::eval( ParseContext *_context ) const
{
ParseContext c1( _context );
ParseContext c2( _context );
// don't evaluate both expressions but return immediately
// if the first one of them succeeds. Otherwise queries like
// ((not exist Blah) or (Blah == 'Foo')) do not work, because
// the evaluation of the second term ends up in a fatal error
// (Simon)
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( c1.type != ParseContext::T_BOOL )
return false;
_context->b = c1.b;
_context->type = ParseContext::T_BOOL;
if ( c1.b )
return true;
if ( !m_pRight->eval( &c2 ) )
return false;
if ( c2.type != ParseContext::T_BOOL )
return false;
_context->b = ( c1.b || c2.b );
_context->type = ParseContext::T_BOOL;
return true;
}
bool ParseTreeAND::eval( ParseContext *_context ) const
{
_context->type = ParseContext::T_BOOL;
ParseContext c1( _context );
ParseContext c2( _context );
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( c1.type != ParseContext::T_BOOL )
return false;
if ( !c1.b )
{
_context->b = false;
return true;
}
if ( !m_pRight->eval( &c2 ) )
return false;
if ( c2.type != ParseContext::T_BOOL )
return false;
_context->b = ( c1.b && c2.b );
return true;
}
bool ParseTreeCALC::eval( ParseContext *_context ) const
{
ParseContext c1( _context );
ParseContext c2( _context );
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( !m_pRight->eval( &c2 ) )
return false;
// Bool extension
if ( c1.type != ParseContext::T_NUM && c1.type != ParseContext::T_DOUBLE && c1.type != ParseContext::T_BOOL )
return false;
// Bool extension
if ( c2.type != ParseContext::T_NUM && c2.type != ParseContext::T_DOUBLE && c2.type != ParseContext::T_BOOL )
return false;
// Bool extension
if ( c1.type == ParseContext::T_BOOL && c2.type == ParseContext::T_BOOL )
return false;
/**
* Make types compatible
*/
if ( c1.type == ParseContext::T_NUM && c2.type == ParseContext::T_DOUBLE )
{
c1.type = ParseContext::T_DOUBLE;
c1.f = (double)c1.i;
}
else if ( c1.type == ParseContext::T_DOUBLE && c2.type == ParseContext::T_NUM )
{
c2.type = ParseContext::T_DOUBLE;
c2.f = (double)c2.i;
}
// Bool extension
else if ( c1.type == ParseContext::T_BOOL && c2.type == ParseContext::T_NUM )
{
c1.type = ParseContext::T_NUM;
if ( c1.b )
c1.i = 1;
else
c1.i = -1;
}
// Bool extension
else if ( c1.type == ParseContext::T_BOOL && c2.type == ParseContext::T_DOUBLE )
{
c1.type = ParseContext::T_DOUBLE;
if ( c1.b )
c1.f = 1.0;
else
c1.f = -1.0;
}
// Bool extension
else if ( c1.type == ParseContext::T_NUM && c2.type == ParseContext::T_BOOL )
{
c2.type = ParseContext::T_NUM;
if ( c2.b )
c2.i = 1;
else
c2.i = -1;
}
// Bool extension
else if ( c1.type == ParseContext::T_DOUBLE && c2.type == ParseContext::T_BOOL )
{
c2.type = ParseContext::T_DOUBLE;
if ( c2.b )
c2.f = 1.0;
else
c2.f = -1.0;
}
_context->type = c1.type;
/**
* Calculate
*/
switch( m_cmd )
{
case 1: /* Add */
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->f = ( c1.f + c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->i = ( c1.i + c2.i );
return true;
}
break;
case 2: /* Sub */
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->f = ( c1.f - c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->i = ( c1.i - c2.i );
return true;
}
break;
case 3: /* Mul */
if ( c1.type == ParseContext::T_DOUBLE )
{
//cout << "Double Mult" << endl;
_context->f = ( c1.f * c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->i = ( c1.i * c2.i );
return true;
}
break;
case 4: /* Div */
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->f = ( c1.f / c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->i = ( c1.i / c2.i );
return true;
}
break;
}
return false;
}
bool ParseTreeCMP::eval( ParseContext *_context ) const
{
//cout << "CMP 1 cmd=" << m_cmd << endl;
ParseContext c1( _context );
ParseContext c2( _context );
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( !m_pRight->eval( &c2 ) )
return false;
/**
* Make types compatible
*/
if ( c1.type == ParseContext::T_NUM && c2.type == ParseContext::T_DOUBLE )
{
c1.type = ParseContext::T_DOUBLE;
c1.f = (double)c1.i;
}
else if ( c1.type == ParseContext::T_DOUBLE && c2.type == ParseContext::T_NUM )
{
c2.type = ParseContext::T_DOUBLE;
c2.f = (double)c2.i;
}
/**
* Compare
*/
_context->type = ParseContext::T_BOOL;
switch( m_cmd )
{
case 1: /* EQ */
case 7: /* EQI */
if ( c1.type != c2.type )
{
_context->b = false;
return true;
}
if ( c1.type == ParseContext::T_STRING )
{
if (m_cmd == 7) {
_context->b = QString::compare(c1.str, c2.str, Qt::CaseInsensitive) == 0;
} else {
_context->b = ( c1.str == c2.str );
}
return true;
}
if ( c1.type == ParseContext::T_BOOL )
{
_context->b = ( c1.b == c2.b );
return true;
}
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->b = ( c1.f == c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->b = ( c1.i == c2.i );
return true;
}
break;
case 2: /* NEQ */
case 8: /* NEQI */
if ( c1.type != c2.type )
{
_context->b = true;
return true;
}
if ( c1.type == ParseContext::T_STRING ) {
if (m_cmd == 8) {
_context->b = QString::compare(c1.str, c2.str, Qt::CaseInsensitive) != 0;
} else {
_context->b = ( c1.str != c2.str );
}
return true;
}
if ( c1.type == ParseContext::T_BOOL )
{
_context->b = ( c1.b != c2.b );
return true;
}
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->b = ( c1.f != c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->b = ( c1.i != c2.i );
return true;
}
break;
case 3: /* GEQ */
if ( c1.type != c2.type )
{
_context->b = false;
return true;
}
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->b = ( c1.f >= c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->b = ( c1.i >= c2.i );
return true;
}
_context->b = false;
return true;
case 4: /* LEQ */
if ( c1.type != c2.type )
{
_context->b = false;
return true;
}
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->b = ( c1.f <= c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->b = ( c1.i <= c2.i );
return true;
}
_context->b = false;
return true;
case 5: /* < */
if ( c1.type != c2.type )
{
_context->b = false;
return true;
}
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->b = ( c1.f < c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->b = ( c1.i < c2.i );
return true;
}
_context->b = false;
return true;
case 6: /* > */
if ( c1.type != c2.type )
{
_context->b = false;
return true;
}
if ( c1.type == ParseContext::T_DOUBLE )
{
_context->b = ( c1.f > c2.f );
return true;
}
if ( c1.type == ParseContext::T_NUM )
{
_context->b = ( c1.i > c2.i );
return true;
}
_context->b = false;
return true;
}
return false;
}
bool ParseTreeNOT::eval( ParseContext *_context ) const
{
ParseContext c1( _context );
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( c1.type != ParseContext::T_BOOL )
return false;
_context->b = !c1.b;
_context->type = ParseContext::T_BOOL;
return true;
}
bool ParseTreeEXIST::eval( ParseContext *_context ) const
{
_context->type = ParseContext::T_BOOL;
QVariant prop = _context->service->property( m_id );
_context->b = prop.isValid();
return true;
}
bool ParseTreeMATCH::eval( ParseContext *_context ) const
{
_context->type = ParseContext::T_BOOL;
ParseContext c1( _context );
ParseContext c2( _context );
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( !m_pRight->eval( &c2 ) )
return false;
if ( c1.type != ParseContext::T_STRING || c2.type != ParseContext::T_STRING )
return false;
_context->b = c2.str.contains( c1.str, m_cs );
return true;
}
bool ParseTreeIN::eval( ParseContext *_context ) const
{
_context->type = ParseContext::T_BOOL;
ParseContext c1( _context );
ParseContext c2( _context );
if ( !m_pLeft->eval( &c1 ) )
return false;
if ( !m_pRight->eval( &c2 ) )
return false;
if ( (c1.type == ParseContext::T_NUM) &&
(c2.type == ParseContext::T_SEQ) &&
((*(c2.seq.begin())).type() == QVariant::Int)) {
QList<QVariant>::ConstIterator it = c2.seq.constBegin();
QList<QVariant>::ConstIterator end = c2.seq.constEnd();
_context->b = false;
for (; it != end; ++it)
if ((*it).type() == QVariant::Int &&
(*it).toInt() == c1.i) {
_context->b = true;
break;
}
return true;
}
if ( c1.type == ParseContext::T_DOUBLE &&
c2.type == ParseContext::T_SEQ &&
(*(c2.seq.begin())).type() == QVariant::Double) {
QList<QVariant>::ConstIterator it = c2.seq.constBegin();
QList<QVariant>::ConstIterator end = c2.seq.constEnd();
_context->b = false;
for (; it != end; ++it)
if ((*it).type() == QVariant::Double &&
(*it).toDouble() == c1.i) {
_context->b = true;
break;
}
return true;
}
if (c1.type == ParseContext::T_STRING && c2.type == ParseContext::T_STR_SEQ)
{
if (false && m_substring) {
_context->b = false;
foreach (const QString &string, c2.strSeq) {
if (string.contains(c1.str, m_cs)) {
_context->b = true;
break;
}
}
} else {
_context->b = c2.strSeq.contains(c1.str, m_cs);
}
return true;
}
return false;
}
bool ParseTreeID::eval( ParseContext *_context ) const
{
QVariant prop = _context->service->property( m_str );
if ( !prop.isValid() )
return false;
if ( prop.type() == QVariant::String )
{
_context->str = prop.toString();
_context->type = ParseContext::T_STRING;
return true;
}
if ( prop.type() == QVariant::Int )
{
_context->i = prop.toInt();
_context->type = ParseContext::T_NUM;
return true;
}
if ( prop.type() == QVariant::Bool )
{
_context->b = prop.toBool();
_context->type = ParseContext::T_BOOL;
return true;
}
if ( prop.type() == QVariant::Double )
{
_context->f = prop.toDouble();
_context->type = ParseContext::T_DOUBLE;
return true;
}
if ( prop.type() == QVariant::List )
{
_context->seq = prop.toList();
_context->type = ParseContext::T_SEQ;
return true;
}
if ( prop.type() == QVariant::StringList )
{
_context->strSeq = prop.toStringList();
_context->type = ParseContext::T_STR_SEQ;
return true;
}
// Value has unknown type
return false;
}
bool ParseTreeMIN2::eval( ParseContext *_context ) const
{
_context->type = ParseContext::T_DOUBLE;
QVariant prop = _context->service->property( m_strId );
if ( !prop.isValid() )
return false;
if ( !_context->initMaxima( m_strId ) )
return false;
QMap<QString,PreferencesMaxima>::Iterator it = _context->maxima.find( m_strId );
if ( it == _context->maxima.end() )
return false;
if ( prop.type() == QVariant::Int && it.value().type == PreferencesMaxima::PM_INT )
{
_context->f = (double)( prop.toInt() - it.value().iMin ) /
(double)(it.value().iMax - it.value().iMin ) * (-2.0) + 1.0;
return true;
}
else if ( prop.type() == QVariant::Double && it.value().type == PreferencesMaxima::PM_DOUBLE )
{
_context->f = ( prop.toDouble() - it.value().fMin ) / (it.value().fMax - it.value().fMin )
* (-2.0) + 1.0;
return true;
}
return false;
}
bool ParseTreeMAX2::eval( ParseContext *_context ) const
{
_context->type = ParseContext::T_DOUBLE;
QVariant prop = _context->service->property( m_strId );
if ( !prop.isValid() )
return false;
// Create extrema
if ( !_context->initMaxima( m_strId ) )
return false;
// Find extrema
QMap<QString,PreferencesMaxima>::Iterator it = _context->maxima.find( m_strId );
if ( it == _context->maxima.end() )
return false;
if ( prop.type() == QVariant::Int && it.value().type == PreferencesMaxima::PM_INT )
{
_context->f = (double)( prop.toInt() - it.value().iMin ) /
(double)(it.value().iMax - it.value().iMin ) * 2.0 - 1.0;
return true;
}
else if ( prop.type() == QVariant::Double && it.value().type == PreferencesMaxima::PM_DOUBLE )
{
_context->f = ( prop.toDouble() - it.value().fMin ) /
(it.value().fMax - it.value().fMin ) * 2.0 - 1.0;
return true;
}
return false;
}
int matchConstraint( const ParseTreeBase *_tree, const KService::Ptr &_service,
const KService::List& _list )
{
// Empty tree matches always
if ( !_tree )
return 1;
QMap<QString,PreferencesMaxima> maxima;
ParseContext c( _service, _list, maxima );
// Error during evaluation ?
if ( !_tree->eval( &c ) )
return -1;
// Did we get a bool ?
if ( c.type != ParseContext::T_BOOL )
return -1;
return ( c.b ? 1 : 0 );
}
bool ParseContext::initMaxima( const QString& _prop )
{
// Is the property known ?
QVariant prop = service->property( _prop );
if ( !prop.isValid() )
return false;
// Numeric ?
if ( prop.type() != QVariant::Int && prop.type() != QVariant::Double )
return false;
// Did we cache the result ?
QMap<QString,PreferencesMaxima>::Iterator it = maxima.find( _prop );
if ( it != maxima.end() )
return ( it.value().type == PreferencesMaxima::PM_DOUBLE ||
it.value().type == PreferencesMaxima::PM_INT );
// Double or Int ?
PreferencesMaxima extrema;
if ( prop.type() == QVariant::Int )
extrema.type = PreferencesMaxima::PM_INVALID_INT;
else
extrema.type = PreferencesMaxima::PM_INVALID_DOUBLE;
// Iterate over all offers
KService::List::ConstIterator oit = offers.begin();
for( ; oit != offers.end(); ++oit )
{
QVariant p = (*oit)->property( _prop );
if ( p.isValid() )
{
// Determine new maximum/minimum
if ( extrema.type == PreferencesMaxima::PM_INVALID_INT )
{
extrema.type = PreferencesMaxima::PM_INT;
extrema.iMin = p.toInt();
extrema.iMax = p.toInt();
}
// Correct existing extrema
else if ( extrema.type == PreferencesMaxima::PM_INT )
{
if ( p.toInt() < extrema.iMin )
extrema.iMin = p.toInt();
if ( p.toInt() > extrema.iMax )
extrema.iMax = p.toInt();
}
// Determine new maximum/minimum
else if ( extrema.type == PreferencesMaxima::PM_INVALID_DOUBLE )
{
extrema.type = PreferencesMaxima::PM_DOUBLE;
extrema.fMin = p.toDouble();
extrema.fMax = p.toDouble();
}
// Correct existing extrema
else if ( extrema.type == PreferencesMaxima::PM_DOUBLE )
{
if ( p.toDouble() < it.value().fMin )
extrema.fMin = p.toDouble();
if ( p.toDouble() > it.value().fMax )
extrema.fMax = p.toDouble();
}
}
}
// Cache the result
maxima.insert( _prop, extrema );
// Did we succeed ?
return ( extrema.type == PreferencesMaxima::PM_DOUBLE ||
extrema.type == PreferencesMaxima::PM_INT );
}
}