|
|
|
/***************************************************************************
|
|
|
|
* This file is part of the KDE project
|
|
|
|
* copyright (C) 2006 by Sebastian Sauer (mail@dipe.org)
|
|
|
|
* copyright (C) 2006 by Sascha Kupper (kusato@kfnv.de)
|
|
|
|
*
|
|
|
|
* This program 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 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
* You should have received a copy of the GNU Library General Public License
|
|
|
|
* along with this program; see the file COPYING. If not, write to
|
|
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#include "macroitem.h"
|
|
|
|
|
|
|
|
#include <kdebug.h>
|
|
|
|
|
|
|
|
using namespace KoMacro;
|
|
|
|
|
|
|
|
namespace KoMacro {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal d-pointer class to be more flexible on future extension of the
|
|
|
|
* functionality without to much risk to break the binary compatibility.
|
|
|
|
*/
|
|
|
|
class MacroItem::Private
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* The @a Action this @a MacroItem has.
|
|
|
|
*/
|
|
|
|
KSharedPtr<Action> action;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The comment this @a MacroItem has.
|
|
|
|
*/
|
|
|
|
TQString comment;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The @a TQMap of @a Variable this @a MacroItem has.
|
|
|
|
*/
|
|
|
|
Variable::Map variables;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* define a @a TQVariant -cast as inline for better performance
|
|
|
|
* @return the casted @a TQVariant by passing a @param variant and its
|
|
|
|
* expected TQVariant::Type @param type.
|
|
|
|
*/
|
|
|
|
inline const TQVariant cast(const TQVariant& variant, TQVariant::Type type) const
|
|
|
|
{
|
|
|
|
// If ok is true the TQVariant v holds our new and to the correct type
|
|
|
|
// casted variant value. If ok is false the as argument passed variant
|
|
|
|
// TQVariant contains the (maybe uncasted string to prevent data-loosing
|
|
|
|
// what would happen if we e.g. would expect an integer and cast it to
|
|
|
|
// an incompatible non-int string) value.
|
|
|
|
bool ok = false;
|
|
|
|
TQVariant v;
|
|
|
|
|
|
|
|
// Try to cast the passed variant to the expected variant-type.
|
|
|
|
switch(type) {
|
|
|
|
case TQVariant::Bool: {
|
|
|
|
const TQString s = variant.toString();
|
|
|
|
ok = (s == "true" || s == "false" || s == "0" || s == "1" || s == "-1");
|
|
|
|
v = TQVariant( variant.toBool(), 0 );
|
|
|
|
} break;
|
|
|
|
case TQVariant::Int: {
|
|
|
|
v = variant.toInt(&ok);
|
|
|
|
// Check if the cast is correct.
|
|
|
|
Q_ASSERT(!ok || v.toString() == variant.toString());
|
|
|
|
} break;
|
|
|
|
case TQVariant::UInt: {
|
|
|
|
v = variant.toUInt(&ok);
|
|
|
|
Q_ASSERT(!ok || v.toString() == variant.toString());
|
|
|
|
} break;
|
|
|
|
case TQVariant::LongLong: {
|
|
|
|
v = variant.toLongLong(&ok);
|
|
|
|
Q_ASSERT(!ok || v.toString() == variant.toString());
|
|
|
|
} break;
|
|
|
|
case TQVariant::ULongLong: {
|
|
|
|
v = variant.toULongLong(&ok);
|
|
|
|
Q_ASSERT(!ok || v.toString() == variant.toString());
|
|
|
|
} break;
|
|
|
|
case TQVariant::Double: {
|
|
|
|
v = variant.toDouble(&ok);
|
|
|
|
Q_ASSERT(!ok || v.toString() == variant.toString());
|
|
|
|
} break;
|
|
|
|
case TQVariant::String: {
|
|
|
|
ok = true; // cast will always be successfully
|
|
|
|
v = variant.toString();
|
|
|
|
} break;
|
|
|
|
default: {
|
|
|
|
// If we got another type we try to let TQt handle it...
|
|
|
|
ok = v.cast(type);
|
|
|
|
kdWarning()<<"MacroItem::Private::cast() Unhandled ok="<<ok<<" type="<<type<<" value="<<v<<endl;
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ok ? v : variant;
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
MacroItem::MacroItem()
|
|
|
|
: KShared()
|
|
|
|
, d( new Private() )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
MacroItem::~MacroItem()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
TQString MacroItem::comment() const
|
|
|
|
{
|
|
|
|
return d->comment;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MacroItem::setComment(const TQString& comment)
|
|
|
|
{
|
|
|
|
d->comment = comment;
|
|
|
|
}
|
|
|
|
|
|
|
|
KSharedPtr<Action> MacroItem::action() const
|
|
|
|
{
|
|
|
|
return d->action;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MacroItem::setAction(KSharedPtr<Action> action)
|
|
|
|
{
|
|
|
|
d->action = action;
|
|
|
|
}
|
|
|
|
|
|
|
|
TQVariant MacroItem::variant(const TQString& name, bool checkaction) const
|
|
|
|
{
|
|
|
|
KSharedPtr<Variable> v = variable(name, checkaction);
|
|
|
|
return v.data() ? v->variant() : TQVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
KSharedPtr<Variable> MacroItem::variable(const TQString& name, bool checkaction) const
|
|
|
|
{
|
|
|
|
if(d->variables.contains(name))
|
|
|
|
return d->variables[name];
|
|
|
|
if(checkaction && d->action.data())
|
|
|
|
return d->action->variable(name);
|
|
|
|
return KSharedPtr<Variable>(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
Variable::Map MacroItem::variables() const
|
|
|
|
{
|
|
|
|
return d->variables;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MacroItem::setVariant(const TQString& name, const TQVariant& variant)
|
|
|
|
{
|
|
|
|
// Let's look if there is an action defined for the variable. If that's
|
|
|
|
// the case, we try to use that action to preserve the type of the variant.
|
|
|
|
KSharedPtr<Variable> actionvariable = d->action ? d->action->variable(name) : KSharedPtr<Variable>(0);
|
|
|
|
|
|
|
|
// If we know the expected type, we try to cast the variant to the expected
|
|
|
|
// type else the variant stays untouched (so, it will stay a string).
|
|
|
|
const TQVariant v = actionvariable.data()
|
|
|
|
? d->cast(variant, actionvariable->variant().type()) // try to cast the variant
|
|
|
|
: variant; // don't cast anything, just leave the string-type...
|
|
|
|
|
|
|
|
// Now let's try to determinate the variable which should be changed.
|
|
|
|
KSharedPtr<Variable> variable = d->variables[name];
|
|
|
|
if(! variable.data()) {
|
|
|
|
// if there exists no such variable yet, create one.
|
|
|
|
kdDebug() << "MacroItem::setVariable() Creating new variable name=" << name << endl;
|
|
|
|
|
|
|
|
variable = KSharedPtr<Variable>( new Variable() );
|
|
|
|
variable->setName(name);
|
|
|
|
d->variables.replace(name, variable);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remember the previous value for the case we like to restore it.
|
|
|
|
const TQVariant oldvar = variable->variant();
|
|
|
|
|
|
|
|
// Set the variable.
|
|
|
|
variable->setVariant(v);
|
|
|
|
|
|
|
|
// Now we inform the referenced action that a variable changed. If
|
|
|
|
// notifyUpdated() returns false, the action rejects the new variable
|
|
|
|
// and we need to restore the previous value.
|
|
|
|
if(d->action && ! d->action->notifyUpdated(this, name)) {
|
|
|
|
kdWarning() << "MacroItem::setVariable() Notify failed for variable name=" << name << endl;
|
|
|
|
variable->setVariant(oldvar);
|
|
|
|
return false; // the action rejected the changed variable whyever...
|
|
|
|
}
|
|
|
|
|
|
|
|
// Job done successfully. The variable is changed to the new value.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
KSharedPtr<Variable> MacroItem::addVariable(const TQString& name, const TQVariant& variant)
|
|
|
|
{
|
|
|
|
Q_ASSERT(! d->variables.contains(name) );
|
|
|
|
// Create a new Variable.
|
|
|
|
KSharedPtr<Variable> variable = KSharedPtr<Variable>( new Variable() );
|
|
|
|
variable->setName(name);
|
|
|
|
|
|
|
|
// Put it into the Variable-map.
|
|
|
|
d->variables.replace(name, variable);
|
|
|
|
|
|
|
|
// Set the variant of the Variable.
|
|
|
|
this->setVariant(name, variant);
|
|
|
|
return variable;
|
|
|
|
}
|