/******************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2011 Thomas Lübking 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, see . *********************************************************************/ #include "anidata_p.h" #include using namespace KWin; static int Gaussian = 46; AniData::AniData() { attribute = AnimationEffect::Opacity; windowType = (NET::WindowTypeMask)0; duration = time = meta = startTime = 0; waitAtSource = false; } AniData::AniData(AnimationEffect::Attribute a, int meta, int ms, const FPx2 &to, QEasingCurve curve, int delay, const FPx2 &from, bool waitAtSource ) { attribute = a; this->from = from; this->to = to; this->curve = curve; duration = ms; time = 0; this->meta = meta; this->waitAtSource = waitAtSource; startTime = AnimationEffect::clock() + delay; } AniData::AniData(const AniData &other) { attribute = other.attribute; from = other.from; to = other.to; time = other.time; duration = other.duration; curve = other.curve; customCurve = other.customCurve; windowType = other.windowType; meta = other.meta; startTime = other.startTime; } static FPx2 fpx2(const QString &s, AnimationEffect::Attribute a) { bool ok; float f1, f2; QStringList floats = s.split(','); f1 = floats.at(0).toFloat(&ok); if (!ok || (f1 < 0.0 && !( a == AnimationEffect::Position || a == AnimationEffect::Translation || a == AnimationEffect::Size || a == AnimationEffect::Rotation)) ) { if (ok) kDebug(1212) << "Invalid value (must not be negative)" << s; return FPx2(); } bool forced_align = false; if ((forced_align = floats.count() < 2)) f2 = f1; else { f2 = floats.at(1).toFloat(&ok); if ( (forced_align = !ok || (f2 < 0.0 && !( a == AnimationEffect::Position || a == AnimationEffect::Translation || a == AnimationEffect::Size || a == AnimationEffect::Rotation))) ) f2 = f1; } if ( forced_align && a >= AnimationEffect::NonFloatBase ) kDebug(1212) << "Generic Animations, WARNING: had to align second dimension of non-onedimensional attribute" << a; return FPx2(f1, f2); } AniData::AniData(const QString &str) // format: WindowMask:Attribute:Meta:Duration:To:Shape:Delay:From { time = 0; duration = 1; // invalidate customCurve = 0; // Linear QStringList animation = str.split(':'); if (animation.count() < 5) return; // at least window type, attribute, metadata, time and target is required windowType = (NET::WindowTypeMask)animation.at(0).toUInt(); if (animation.at(1) == "Opacity") attribute = AnimationEffect::Opacity; else if (animation.at(1) == "Brightness") attribute = AnimationEffect::Brightness; else if (animation.at(1) == "Saturation") attribute = AnimationEffect::Saturation; else if (animation.at(1) == "Scale") attribute = AnimationEffect::Scale; else if (animation.at(1) == "Translation") attribute = AnimationEffect::Translation; else if (animation.at(1) == "Rotation") attribute = AnimationEffect::Rotation; else if (animation.at(1) == "Position") attribute = AnimationEffect::Position; else if (animation.at(1) == "Size") attribute = AnimationEffect::Size; else if (animation.at(1) == "Clip") attribute = AnimationEffect::Clip; else { kDebug(1212) << "Invalid attribute" << animation.at(1); return; } meta = animation.at(2).toUInt(); bool ok; duration = animation.at(3).toInt(&ok); if (!ok || duration < 0) { kDebug(1212) << "Invalid duration" << animation.at(3); duration = 0; return; } to = fpx2(animation.at(4), attribute); if (animation.count() > 5) { customCurve = animation.at(5).toInt(); if (customCurve < QEasingCurve::Custom) curve.setType((QEasingCurve::Type)customCurve); else if (customCurve == Gaussian) curve.setCustomType(AnimationEffect::qecGaussian); else kDebug(1212) << "Unknown curve type" << customCurve; // remains default, ie. linear if (animation.count() > 6) { int t = animation.at(6).toInt(); if (t < 0) kDebug(1212) << "Delay can not be negative" << animation.at(6); else time = t; if (animation.count() > 7) from = fpx2(animation.at(7), attribute); } } if (!(from.isValid() || to.isValid())) { duration = -1; // invalidate return; } } QList AniData::list(const QString &str) { QList newList; QStringList list = str.split(';', QString::SkipEmptyParts); foreach (const QString &astr, list) { newList << AniData(astr); if (newList.last().duration < 0) newList.removeLast(); } return newList; } QString AniData::toString() const { QString ret = QString::number((uint)windowType) + ':'; switch (attribute) { case AnimationEffect::Opacity: ret += "Opacity:"; break; case AnimationEffect::Brightness: ret += "Brightness:"; break; case AnimationEffect::Saturation: ret += "Saturation:"; break; case AnimationEffect::Scale: ret += "Scale:"; break; case AnimationEffect::Translation: ret += "Translation:"; break; case AnimationEffect::Rotation: ret += "Rotation:"; break; case AnimationEffect::Position: ret += "Position:"; break; case AnimationEffect::Size: ret += "Size:"; break; case AnimationEffect::Clip: ret += "Clip:"; break; default: ret += ':'; } ret += QString::number(meta) + ':' + QString::number(duration) + ':' + to.toString() + ':' + QString::number(customCurve) + ':' + QString::number(time) + ':' + from.toString(); return ret; }