summaryrefslogtreecommitdiffstats
path: root/tdeio/tdeio/tdefilemetainfo.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'tdeio/tdeio/tdefilemetainfo.cpp')
-rw-r--r--tdeio/tdeio/tdefilemetainfo.cpp1859
1 files changed, 1859 insertions, 0 deletions
diff --git a/tdeio/tdeio/tdefilemetainfo.cpp b/tdeio/tdeio/tdefilemetainfo.cpp
new file mode 100644
index 000000000..c1389695d
--- /dev/null
+++ b/tdeio/tdeio/tdefilemetainfo.cpp
@@ -0,0 +1,1859 @@
+/*
+ * This file is part of the KDE libraries
+ * Copyright (C) 2001-2002 Rolf Magnus <ramagnus@kde.org>
+ * Copyright (C) 2001-2002 Carsten Pfeiffer <pfeiffer@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 version 2.0.
+ *
+ * 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.
+ *
+ * $Id$
+ */
+
+#include <assert.h>
+
+#include <tqshared.h>
+#include <tqdict.h>
+
+#include <ktrader.h>
+#include <kstaticdeleter.h>
+#include <tdeparts/componentfactory.h>
+#include <kuserprofile.h>
+#include <kdebug.h>
+#include <kmimetype.h>
+#include <kdatastream.h> // needed for serialization of bool
+#include <tdelocale.h>
+#include <tdeio/global.h>
+
+#include "tdefilemetainfo.h"
+
+// shared data of a KFileMetaInfoItem
+class KFileMetaInfoItem::Data : public TQShared
+{
+public:
+ Data( const KFileMimeTypeInfo::ItemInfo* mti, const TQString& _key,
+ const TQVariant& _value )
+ : TQShared(),
+ mimeTypeInfo( mti ),
+ key( _key ),
+ value( _value ),
+ dirty( false ),
+ added( false ),
+ removed( false )
+ {}
+
+ // we use this one for the streaming operators
+ Data() : mimeTypeInfo( 0L )
+ {}
+
+ ~Data()
+ {
+ if ( this == null ) // only the null item owns its mimeTypeInfo
+ delete mimeTypeInfo;
+ }
+
+ const KFileMimeTypeInfo::ItemInfo* mimeTypeInfo;
+ // mimeTypeInfo has the key, too, but only for non-variable ones
+ TQString key;
+ TQVariant value;
+ bool dirty :1;
+ bool added :1;
+ bool removed :1;
+
+ static Data* null;
+ static Data* makeNull();
+};
+
+//this is our null data
+KFileMetaInfoItem::Data* KFileMetaInfoItem::Data::null = 0L;
+static KStaticDeleter<KFileMetaInfoItem::Data> sd_KFileMetaInfoItemData;
+
+KFileMetaInfoItem::Data* KFileMetaInfoItem::Data::makeNull()
+{
+ if (!null)
+ {
+ // We deliberately do not reset "null" after it has been destroyed!
+ // Otherwise we will run into problems later in ~KFileMetaInfoItem
+ // where the d-pointer is compared against null.
+
+ KFileMimeTypeInfo::ItemInfo* info = new KFileMimeTypeInfo::ItemInfo();
+ null = new Data(info, TQString::null, TQVariant());
+ sd_KFileMetaInfoItemData.setObject( null );
+ }
+ return null;
+}
+
+KFileMetaInfoItem::KFileMetaInfoItem( const KFileMimeTypeInfo::ItemInfo* mti,
+ const TQString& key, const TQVariant& value )
+ : d( new Data( mti, key, value ) )
+{
+}
+
+KFileMetaInfoItem::KFileMetaInfoItem( const KFileMetaInfoItem& item )
+{
+ // operator= does everything that's necessary
+ d = Data::makeNull();
+ *this = item;
+}
+
+KFileMetaInfoItem::KFileMetaInfoItem()
+{
+ d = Data::makeNull();
+}
+
+KFileMetaInfoItem::~KFileMetaInfoItem()
+{
+ deref();
+}
+
+const KFileMetaInfoItem& KFileMetaInfoItem::operator=
+ (const KFileMetaInfoItem & item )
+{
+ if (d != item.d)
+ {
+ // first deref the old one
+ deref();
+ d = item.d;
+ // and now ref the new one
+ ref();
+ }
+
+ return *this;
+}
+
+bool KFileMetaInfoItem::setValue( const TQVariant& value )
+{
+ // We don't call makeNull here since it isn't necassery, see deref()
+ if ( d == Data::null ) return false;
+
+ if ( ! (d->mimeTypeInfo->attributes() & KFileMimeTypeInfo::Modifiable ) ||
+ ! (value.canCast(d->mimeTypeInfo->type())))
+ {
+ kdDebug(7033) << "setting the value of " << key() << "failed\n";
+ return false;
+ }
+
+// kdDebug(7033) << key() << ".setValue()\n";
+
+ if ( d->value == value )
+ return true;
+
+ d->dirty = true;
+ d->value = value;
+ // If we don't cast (and test for canCast in the above if), TQVariant is
+ // very picky about types (e.g. TQString vs. TQCString or int vs. uint)
+ d->value.cast(d->mimeTypeInfo->type());
+
+ return true;
+}
+
+bool KFileMetaInfoItem::isRemoved() const
+{
+ return d->removed;
+}
+
+TQString KFileMetaInfoItem::key() const
+{
+ return d->key;
+}
+
+TQString KFileMetaInfoItem::translatedKey() const
+{
+ // are we a variable key?
+ if (d->mimeTypeInfo->key().isNull())
+ {
+ // then try if we have luck with i18n()
+ return i18n(d->key.utf8());
+ }
+
+ return d->mimeTypeInfo->translatedKey();
+}
+
+const TQVariant& KFileMetaInfoItem::value() const
+{
+ return d->value;
+}
+
+TQString KFileMetaInfoItem::string( bool mangle ) const
+{
+ return d->mimeTypeInfo->string(d->value, mangle);
+}
+
+TQVariant::Type KFileMetaInfoItem::type() const
+{
+ return d->mimeTypeInfo->type();
+}
+
+uint KFileMetaInfoItem::unit() const
+{
+ return d->mimeTypeInfo->unit();
+}
+
+bool KFileMetaInfoItem::isModified() const
+{
+ return d->dirty;
+}
+
+TQString KFileMetaInfoItem::prefix() const
+{
+ return d->mimeTypeInfo->prefix();
+}
+
+TQString KFileMetaInfoItem::suffix() const
+{
+ return d->mimeTypeInfo->suffix();
+}
+
+uint KFileMetaInfoItem::hint() const
+{
+ return d->mimeTypeInfo->hint();
+}
+
+uint KFileMetaInfoItem::attributes() const
+{
+ return d->mimeTypeInfo->attributes();
+}
+
+bool KFileMetaInfoItem::isEditable() const
+{
+ return d->mimeTypeInfo->attributes() & KFileMimeTypeInfo::Modifiable;
+}
+
+bool KFileMetaInfoItem::isValid() const
+{
+ // We don't call makeNull here since it isn't necassery:
+ // If d is equal to null it means that null is initialized already.
+ // null is 0L when it hasn't been initialized and d is never 0L.
+ return d != Data::null;
+}
+
+void KFileMetaInfoItem::setAdded()
+{
+ d->added = true;
+}
+
+void KFileMetaInfoItem::setRemoved()
+{
+ d->removed = true;
+}
+
+void KFileMetaInfoItem::ref()
+{
+ if (d != Data::null) d->ref();
+}
+
+void KFileMetaInfoItem::deref()
+{
+ // We don't call makeNull here since it isn't necassery:
+ // If d is equal to null it means that null is initialized already.
+ // null is 0L when it hasn't been initialized and d is never 0L.
+ if ((d != Data::null) && d->deref())
+ {
+// kdDebug(7033) << "item " << d->key
+// << " is finally deleted\n";
+ delete d;
+ d = 0;
+ }
+}
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+// shared data of a KFileMetaInfo
+class KFileMetaInfo::Data : public TQShared
+{
+public:
+ Data(const KURL& _url, uint _what)
+ : TQShared(),
+ url(_url),
+ what(_what),
+ mimeTypeInfo( 0L )
+ {}
+
+ // wee use this one for the streaming operators
+ Data() {};
+
+ KURL url;
+ uint what;
+ TQMap<TQString, KFileMetaInfoGroup> groups;
+ const KFileMimeTypeInfo* mimeTypeInfo;
+ TQStringList removedGroups;
+
+ static Data* null;
+ static Data* makeNull();
+
+};
+
+KFileMetaInfo::KFileMetaInfo( const TQString& path, const TQString& mimeType,
+ uint what )
+{
+ KURL u;
+
+ u.setPath(path);
+ init(u, mimeType, what);
+}
+
+KFileMetaInfo::KFileMetaInfo( const KURL& url, const TQString& mimeType,
+ uint what )
+{
+ init(url, mimeType, what);
+}
+
+void KFileMetaInfo::init( const KURL& url, const TQString& mimeType,
+ uint what )
+{
+ d = new Data( url, what );
+
+ TQString mT;
+ if (mimeType.isEmpty())
+ mT = KMimeType::findByURL(url)->name();
+ else
+ mT = mimeType;
+
+ // let's "share our property"
+ KFileMetaInfo item(*this);
+
+ //kdDebug() << k_funcinfo << mT << " " << url << endl;
+
+ d->mimeTypeInfo = KFileMetaInfoProvider::self()->mimeTypeInfo( mT, url.protocol() );
+ if ( d->mimeTypeInfo )
+ {
+ //kdDebug(7033) << "Found mimetype info for " << mT /* or protocol*/ << endl;
+ KFilePlugin *p = plugin();
+ Q_ASSERT( p );
+ if ( p && !p->readInfo( item, what) )
+ {
+ deref();
+ d = Data::makeNull();
+ }
+ }
+ else
+ {
+// kdDebug(7033) << "No mimetype info for " << mimeType << endl;
+ deref();
+ d = Data::makeNull();
+ }
+}
+
+KFileMetaInfo::KFileMetaInfo( const KFileMetaInfo& original )
+{
+ // operator= does everything that's necessary
+ d = Data::makeNull();
+ *this = original;
+}
+
+KFileMetaInfo::KFileMetaInfo()
+{
+ d = Data::makeNull();
+}
+
+KFileMetaInfo::~KFileMetaInfo()
+{
+ deref();
+}
+
+TQStringList KFileMetaInfo::supportedGroups() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->supportedGroups();
+}
+
+TQStringList KFileMetaInfo::supportedKeys() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->supportedKeys();
+}
+
+TQStringList KFileMetaInfo::groups() const
+{
+ TQStringList list;
+ TQMapConstIterator<TQString, KFileMetaInfoGroup> it = d->groups.begin();
+ for ( ; it != d->groups.end(); ++it )
+ list += (*it).name();
+
+ return list;
+}
+
+TQStringList KFileMetaInfo::editableGroups() const
+{
+ TQStringList list;
+ TQStringList supported = supportedGroups();
+ TQStringList::ConstIterator it = supported.begin();
+ for ( ; it != supported.end(); ++it ) {
+ const KFileMimeTypeInfo::GroupInfo * groupInfo = d->mimeTypeInfo->groupInfo( *it );
+ if ( groupInfo && groupInfo->attributes() &
+ (KFileMimeTypeInfo::Addable | KFileMimeTypeInfo::Removable) )
+ list.append( *it );
+ }
+
+ return list;
+}
+
+TQStringList KFileMetaInfo::preferredGroups() const
+{
+ assert(isValid());
+ TQStringList list = groups();
+ TQStringList newlist;
+ TQStringList preferred = d->mimeTypeInfo->preferredGroups();
+ TQStringList::Iterator pref;
+
+ // move all keys from the preferred groups that are in our list to a new list
+ for ( pref = preferred.begin(); pref != preferred.end(); ++pref )
+ {
+ TQStringList::Iterator group = list.find(*pref);
+ if ( group != list.end() )
+ {
+ newlist.append( *group );
+ list.remove(group);
+ }
+ }
+
+ // now the old list only contains the non-preferred items, so we
+ // add the remaining ones to newlist
+ newlist += list;
+
+ return newlist;
+}
+
+TQStringList KFileMetaInfo::preferredKeys() const
+{
+ TQStringList newlist;
+
+ TQStringList list = preferredGroups();
+ for (TQStringList::Iterator git = list.begin(); git != list.end(); ++git)
+ {
+ newlist += d->groups[*git].preferredKeys();
+ }
+
+ return newlist;
+}
+
+KFileMetaInfoGroup KFileMetaInfo::group(const TQString& key) const
+{
+ TQMapIterator<TQString,KFileMetaInfoGroup> it = d->groups.find( key );
+ if ( it != d->groups.end() )
+ return it.data();
+ else
+ return KFileMetaInfoGroup();
+}
+
+bool KFileMetaInfo::addGroup( const TQString& name )
+{
+ assert(isValid());
+ if ( d->mimeTypeInfo->supportedGroups().contains(name) &&
+ ! d->groups.contains(name) )
+ {
+ KFileMetaInfoGroup group( name, d->mimeTypeInfo );
+
+ // add all the items that can't be added by the user later
+ const KFileMimeTypeInfo::GroupInfo* ginfo = d->mimeTypeInfo->groupInfo(name);
+ Q_ASSERT(ginfo);
+ if (!ginfo) return false;
+
+ TQStringList keys = ginfo->supportedKeys();
+ for (TQStringList::Iterator it = keys.begin(); it != keys.end(); ++it)
+ {
+ const KFileMimeTypeInfo::ItemInfo* iteminfo = ginfo->itemInfo(*it);
+ Q_ASSERT(ginfo);
+ if (!iteminfo) return false;
+
+ if ( !(iteminfo->attributes() & KFileMimeTypeInfo::Addable) &&
+ (iteminfo->attributes() & KFileMimeTypeInfo::Modifiable))
+ {
+ // append it now or never
+ group.appendItem(iteminfo->key(), TQVariant());
+ }
+
+ }
+
+ d->groups.insert(name, group);
+ group.setAdded();
+ return true;
+ }
+
+ return false;
+}
+
+bool KFileMetaInfo::removeGroup( const TQString& name )
+{
+ TQMapIterator<TQString, KFileMetaInfoGroup> it = d->groups.find(name);
+ if ( (it==d->groups.end()) ||
+ !((*it).attributes() & KFileMimeTypeInfo::Removable))
+ return false;
+
+ d->groups.remove(it);
+ d->removedGroups.append(name);
+ return true;
+}
+
+TQStringList KFileMetaInfo::removedGroups()
+{
+ return d->removedGroups;
+}
+
+const KFileMetaInfo& KFileMetaInfo::operator= (const KFileMetaInfo& info )
+{
+ if (d != info.d)
+ {
+ deref();
+ // first deref the old one
+ d = info.d;
+ // and now ref the new one
+ ref();
+ }
+ return *this;
+}
+
+bool KFileMetaInfo::isValid() const
+{
+ // We don't call makeNull here since it isn't necassery, see deref()
+ return d != Data::null;
+}
+
+bool KFileMetaInfo::isEmpty() const
+{
+ for (TQMapIterator<TQString, KFileMetaInfoGroup> it = d->groups.begin();
+ it!=d->groups.end(); ++it)
+ if (!(*it).isEmpty())
+ return false;
+ return true;
+}
+
+bool KFileMetaInfo::applyChanges()
+{
+ return applyChanges( path() );
+}
+
+bool KFileMetaInfo::applyChanges( const TQString& path )
+{
+ bool doit = false;
+
+// kdDebug(7033) << "KFileMetaInfo::applyChanges()\n";
+
+ // look up if we need to write to the file
+ TQMapConstIterator<TQString, KFileMetaInfoGroup> it;
+ for (it = d->groups.begin(); it!=d->groups.end() && !doit; ++it)
+ {
+ if ( (*it).isModified() )
+ doit = true;
+
+ else
+ {
+ TQStringList keys = it.data().keys();
+ for (TQStringList::Iterator it2 = keys.begin(); it2!=keys.end(); ++it2)
+ {
+ if ( (*it)[*it2].isModified() )
+ {
+ doit = true;
+ break;
+ }
+ }
+ }
+ }
+
+ if (!doit)
+ {
+ kdDebug(7033) << "Don't need to write, nothing changed\n";
+ return true;
+ }
+
+ KFilePlugin* p = plugin();
+ if (!p) return false;
+
+// kdDebug(7033) << "Ok, trying to write the info\n";
+
+ KURL savedURL = url();
+ d->url = KURL();
+ d->url.setPath( path );
+
+ bool ret = p->writeInfo(*this);
+
+ d->url = savedURL;
+ return ret;
+}
+
+KFilePlugin * KFileMetaInfo::plugin() const
+{
+ assert(isValid());
+ KFileMetaInfoProvider* prov = KFileMetaInfoProvider::self();
+ return prov->plugin( d->mimeTypeInfo->mimeType(), d->url.protocol() );
+}
+
+TQString KFileMetaInfo::mimeType() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->mimeType();
+}
+
+bool KFileMetaInfo::contains(const TQString& key) const
+{
+ TQStringList glist = groups();
+ for (TQStringList::Iterator it = glist.begin(); it != glist.end(); ++it)
+ {
+ KFileMetaInfoGroup g = d->groups[*it];
+ if (g.contains(key)) return true;
+ }
+ return false;
+}
+
+bool KFileMetaInfo::containsGroup(const TQString& key) const
+{
+ return groups().contains(key);
+}
+
+KFileMetaInfoItem KFileMetaInfo::item( const TQString& key) const
+{
+ TQStringList groups = preferredGroups();
+ for (TQStringList::Iterator it = groups.begin(); it != groups.end(); ++it)
+ {
+ KFileMetaInfoItem i = d->groups[*it][key];
+ if (i.isValid()) return i;
+ }
+ return KFileMetaInfoItem();
+}
+
+KFileMetaInfoItem KFileMetaInfo::item(const KFileMetaInfoItem::Hint hint) const
+{
+ TQStringList groups = preferredGroups();
+ TQStringList::ConstIterator it;
+ for (it = groups.begin(); it != groups.end(); ++it)
+ {
+ KFileMetaInfoItem i = d->groups[*it].item(hint);
+ if (i.isValid()) return i;
+ }
+ return KFileMetaInfoItem();
+}
+
+KFileMetaInfoItem KFileMetaInfo::saveItem( const TQString& key,
+ const TQString& preferredGroup,
+ bool createGroup )
+{
+ assert(isValid());
+ // try the preferred groups first
+ if ( !preferredGroup.isEmpty() ) {
+ TQMapIterator<TQString,KFileMetaInfoGroup> it =
+ d->groups.find( preferredGroup );
+
+ // try to create the preferred group, if necessary
+ if ( it == d->groups.end() && createGroup ) {
+ const KFileMimeTypeInfo::GroupInfo *groupInfo =
+ d->mimeTypeInfo->groupInfo( preferredGroup );
+ if ( groupInfo && groupInfo->supportedKeys().contains( key ) ) {
+ if ( addGroup( preferredGroup ) )
+ it = d->groups.find( preferredGroup );
+ }
+ }
+
+ if ( it != d->groups.end() ) {
+ KFileMetaInfoItem item = it.data().addItem( key );
+ if ( item.isValid() )
+ return item;
+ }
+ }
+
+ TQStringList groups = preferredGroups();
+
+ KFileMetaInfoItem item;
+
+ TQStringList::ConstIterator groupIt = groups.begin();
+ for ( ; groupIt != groups.end(); ++groupIt )
+ {
+ TQMapIterator<TQString,KFileMetaInfoGroup> it = d->groups.find( *groupIt );
+ if ( it != d->groups.end() )
+ {
+ KFileMetaInfoGroup group = it.data();
+ item = findEditableItem( group, key );
+ if ( item.isValid() )
+ return item;
+ }
+ else // not existant -- try to create the group
+ {
+ const KFileMimeTypeInfo::GroupInfo *groupInfo =
+ d->mimeTypeInfo->groupInfo( *groupIt );
+ if ( groupInfo && groupInfo->supportedKeys().contains( key ) )
+ {
+ if ( addGroup( *groupIt ) )
+ {
+ KFileMetaInfoGroup group = d->groups[*groupIt];
+ KFileMetaInfoItem item = group.addItem( key );
+ if ( item.isValid() )
+ return item;
+// else ### add when removeGroup() is implemented :)
+// removeGroup( *groupIt ); // couldn't add item -> remove
+ }
+ }
+ }
+ }
+
+ // finally check for variable items
+
+ return item;
+}
+
+KFileMetaInfoItem KFileMetaInfo::findEditableItem( KFileMetaInfoGroup& group,
+ const TQString& key )
+{
+ assert(isValid());
+ KFileMetaInfoItem item = group.addItem( key );
+ if ( item.isValid() && item.isEditable() )
+ return item;
+
+ if ( (d->mimeTypeInfo->groupInfo( group.name() )->attributes() & KFileMimeTypeInfo::Addable) )
+ return item;
+
+ return KFileMetaInfoItem();
+}
+
+KFileMetaInfoGroup KFileMetaInfo::appendGroup(const TQString& name)
+{
+ assert(isValid());
+ if ( d->mimeTypeInfo->supportedGroups().contains(name) &&
+ ! d->groups.contains(name) )
+ {
+ KFileMetaInfoGroup group( name, d->mimeTypeInfo );
+ d->groups.insert(name, group);
+ return group;
+ }
+
+ else {
+ kdWarning(7033) << "Someone's trying to add a KFileMetaInfoGroup which is not supported or already existing: " << name << endl;
+ return KFileMetaInfoGroup();
+ }
+}
+
+TQString KFileMetaInfo::path() const
+{
+ return d->url.isLocalFile() ? d->url.path() : TQString::null;
+}
+
+KURL KFileMetaInfo::url() const
+{
+ return d->url;
+}
+
+void KFileMetaInfo::ref()
+{
+ if (d != Data::null) d->ref();
+
+}
+
+void KFileMetaInfo::deref()
+{
+ // We don't call makeNull here since it isn't necassery:
+ // If d is equal to null it means that null is initialized already.
+ // null is 0L when it hasn't been initialized and d is never 0L.
+ if ((d != Data::null) && d->deref())
+ {
+// kdDebug(7033) << "metainfo object for " << d->url.path << " is finally deleted\n";
+ delete d;
+ d = 0;
+ }
+
+}
+
+
+KFileMetaInfo::Data* KFileMetaInfo::Data::null = 0L;
+static KStaticDeleter<KFileMetaInfo::Data> sd_KFileMetaInfoData;
+
+KFileMetaInfo::Data* KFileMetaInfo::Data::makeNull()
+{
+ if (!null)
+ // We deliberately do not reset "null" after it has been destroyed!
+ // Otherwise we will run into problems later in ~KFileMetaInfoItem
+ // where the d-pointer is compared against null.
+ null = sd_KFileMetaInfoData.setObject( new KFileMetaInfo::Data(KURL(), 0) );
+ return null;
+}
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+KFilePlugin::KFilePlugin( TQObject *parent, const char *name,
+ const TQStringList& /*args*/)
+ : TQObject( parent, name )
+{
+// kdDebug(7033) << "loaded a plugin for " << name << endl;
+}
+
+KFilePlugin::~KFilePlugin()
+{
+// kdDebug(7033) << "unloaded a plugin for " << name() << endl;
+}
+
+KFileMimeTypeInfo * KFilePlugin::addMimeTypeInfo( const TQString& mimeType )
+{
+ return KFileMetaInfoProvider::self()->addMimeTypeInfo( mimeType );
+}
+
+void KFilePlugin::virtual_hook( int, void* )
+{ /*BASE::virtual_hook( id, data );*/ }
+
+
+KFileMimeTypeInfo::GroupInfo* KFilePlugin::addGroupInfo(KFileMimeTypeInfo* info,
+ const TQString& key, const TQString& translatedKey) const
+{
+ return info->addGroupInfo(key, translatedKey);
+}
+
+void KFilePlugin::setAttributes(KFileMimeTypeInfo::GroupInfo* gi, uint attr) const
+{
+ gi->m_attr = attr;
+}
+
+void KFilePlugin::addVariableInfo(KFileMimeTypeInfo::GroupInfo* gi,
+ TQVariant::Type type, uint attr) const
+{
+ gi->addVariableInfo(type, attr);
+}
+
+KFileMimeTypeInfo::ItemInfo* KFilePlugin::addItemInfo(KFileMimeTypeInfo::GroupInfo* gi,
+ const TQString& key,
+ const TQString& translatedKey,
+ TQVariant::Type type)
+{
+ return gi->addItemInfo(key, translatedKey, type);
+}
+
+void KFilePlugin::setAttributes(KFileMimeTypeInfo::ItemInfo* item, uint attr)
+{
+ item->m_attr = attr;
+}
+
+void KFilePlugin::setHint(KFileMimeTypeInfo::ItemInfo* item, uint hint)
+{
+ item->m_hint = hint;
+}
+
+void KFilePlugin::setUnit(KFileMimeTypeInfo::ItemInfo* item, uint unit)
+{
+ item->m_unit = unit;
+ // set prefix and suffix
+ switch (unit)
+ {
+ case KFileMimeTypeInfo::Seconds:
+ item->m_suffix = i18n("s"); break;
+
+ case KFileMimeTypeInfo::MilliSeconds:
+ item->m_suffix = i18n("ms"); break;
+
+ case KFileMimeTypeInfo::BitsPerSecond:
+ item->m_suffix = i18n("bps"); break;
+
+ case KFileMimeTypeInfo::Pixels:
+ item->m_suffix = i18n("pixels"); break;
+
+ case KFileMimeTypeInfo::Inches:
+ item->m_suffix = i18n("in"); break;
+
+ case KFileMimeTypeInfo::Centimeters:
+ item->m_suffix = i18n("cm"); break;
+
+ case KFileMimeTypeInfo::Bytes:
+ item->m_suffix = i18n("B"); break;
+
+ case KFileMimeTypeInfo::KiloBytes:
+ item->m_suffix = i18n("KB"); break;
+
+ case KFileMimeTypeInfo::FramesPerSecond:
+ item->m_suffix = i18n("fps"); break;
+
+ case KFileMimeTypeInfo::DotsPerInch:
+ item->m_suffix = i18n("dpi"); break;
+
+ case KFileMimeTypeInfo::BitsPerPixel:
+ item->m_suffix = i18n("bpp"); break;
+
+ case KFileMimeTypeInfo::Hertz:
+ item->m_suffix = i18n("Hz"); break;
+
+ case KFileMimeTypeInfo::Millimeters:
+ item->m_suffix = i18n("mm");
+ }
+}
+
+void KFilePlugin::setPrefix(KFileMimeTypeInfo::ItemInfo* item, const TQString& prefix)
+{
+ item->m_prefix = prefix;
+}
+
+void KFilePlugin::setSuffix(KFileMimeTypeInfo::ItemInfo* item, const TQString& suffix)
+{
+ item->m_suffix = suffix;
+}
+
+KFileMetaInfoGroup KFilePlugin::appendGroup(KFileMetaInfo& info, const TQString& key)
+{
+ return info.appendGroup(key);
+}
+
+void KFilePlugin::appendItem(KFileMetaInfoGroup& group, const TQString& key, TQVariant value)
+{
+ group.appendItem(key, value);
+}
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+
+KFileMetaInfoProvider * KFileMetaInfoProvider::s_self;
+static KStaticDeleter<KFileMetaInfoProvider> sd;
+
+KFileMetaInfoProvider * KFileMetaInfoProvider::self()
+{
+ if ( !s_self )
+ s_self = sd.setObject( s_self, new KFileMetaInfoProvider() );
+
+ return s_self;
+}
+
+KFileMetaInfoProvider::KFileMetaInfoProvider()
+{
+ m_plugins.setAutoDelete( true );
+}
+
+KFileMetaInfoProvider::~KFileMetaInfoProvider()
+{
+ m_plugins.clear();
+ sd.setObject( 0 );
+}
+
+KFilePlugin* KFileMetaInfoProvider::loadPlugin( const TQString& mimeType, const TQString& protocol )
+{
+ //kdDebug() << "loadPlugin: mimeType=" << mimeType << " protocol=" << protocol << endl;
+ // Currently the idea is: either the mimetype is set or the protocol, but not both.
+ // We need PNG fileinfo, and trash: fileinfo, but not "PNG in the trash".
+ TQString queryMimeType, query;
+ if ( !mimeType.isEmpty() ) {
+ query = "(not exist [X-TDE-Protocol])";
+ queryMimeType = mimeType;
+ } else {
+ query = TQString::fromLatin1( "[X-TDE-Protocol] == '%1'" ).arg(protocol);
+ // querying for a protocol: we have no mimetype, so we need to use KFilePlugin as one
+ queryMimeType = "KFilePlugin";
+ // hopefully using KFilePlugin as genericMimeType too isn't a problem
+ }
+ const TDETrader::OfferList offers = TDETrader::self()->query( queryMimeType, "KFilePlugin", query, TQString::null );
+ if ( offers.isEmpty() )
+ return 0;
+ KService::Ptr service = *(offers.begin());
+ Q_ASSERT( service && service->isValid() );
+ if ( !service || !service->isValid() )
+ return 0;
+
+ KFilePlugin* plugin = KParts::ComponentFactory::createInstanceFromService<KFilePlugin>
+ ( service, TQT_TQOBJECT(this), mimeType.local8Bit() );
+ if (!plugin)
+ kdWarning(7033) << "error loading the plugin from " << service->desktopEntryPath() << endl;
+
+ return plugin;
+}
+
+KFilePlugin* KFileMetaInfoProvider::loadAndRegisterPlugin( const TQString& mimeType, const TQString& protocol )
+{
+ Q_ASSERT( m_pendingMimetypeInfos.isEmpty() );
+ m_pendingMimetypeInfos.clear();
+
+ KFilePlugin* plugin = loadPlugin( mimeType, protocol );
+ if ( !plugin ) {
+ // No plugin found. Remember that, to save time.
+ m_plugins.insert( protocol.isEmpty() ? mimeType : protocol, new CachedPluginInfo );
+ return 0;
+ }
+
+ if ( !protocol.isEmpty() ) {
+ // Protocol-metainfo: only one entry
+ Q_ASSERT( m_pendingMimetypeInfos.count() == 1 );
+ KFileMimeTypeInfo* info = m_pendingMimetypeInfos[ protocol ];
+ Q_ASSERT( info );
+ m_plugins.insert( protocol, new CachedPluginInfo( plugin, info, true ) );
+ } else {
+ // Mimetype-metainfo: the plugin can register itself for multiple mimetypes, remember them all
+ bool first = true;
+ TQDictIterator<KFileMimeTypeInfo> it( m_pendingMimetypeInfos );
+ for( ; it.current(); ++it ) {
+ KFileMimeTypeInfo* info = it.current();
+ m_plugins.insert( it.currentKey(), new CachedPluginInfo( plugin, info, first ) );
+ first = false;
+ }
+ // Hopefully the above includes the mimetype we asked for!
+ if ( m_pendingMimetypeInfos.find( mimeType ) == 0 )
+ kdWarning(7033) << plugin->className() << " was created for " << mimeType << " but doesn't call addMimeTypeInfo for it!" << endl;
+ }
+ m_pendingMimetypeInfos.clear();
+ return plugin;
+}
+
+KFilePlugin * KFileMetaInfoProvider::plugin(const TQString& mimeType)
+{
+ return plugin( mimeType, TQString::null );
+}
+
+KFilePlugin * KFileMetaInfoProvider::plugin(const TQString& mimeType, const TQString& protocol)
+{
+ //kdDebug(7033) << "plugin() : looking for plugin for protocol=" << protocol << " mimeType=" << mimeType << endl;
+
+ if ( !protocol.isEmpty() ) {
+ CachedPluginInfo *cache = m_plugins.find( protocol );
+ if ( cache && cache->plugin ) {
+ return cache->plugin;
+ }
+ if ( !cache ) {
+ KFilePlugin* plugin = loadAndRegisterPlugin( TQString::null, protocol );
+ if ( plugin )
+ return plugin;
+ }
+ }
+
+ CachedPluginInfo *cache = m_plugins.find( mimeType );
+ if ( cache ) {
+ return cache->plugin;
+ }
+
+ KFilePlugin* plugin = loadAndRegisterPlugin( mimeType, TQString::null );
+
+#if 0
+ kdDebug(7033) << "currently loaded plugins:\n";
+
+ TQDictIterator<CachedPluginInfo> it( m_plugins );
+ for( ; it.current(); ++it ) {
+ CachedPluginInfo* cache = it.current();
+ kdDebug(7033)
+ << it.currentKey() // mimetype or protocol
+ << " : " << (cache->plugin ? cache->plugin->className() : "(no plugin)") << endl; // plugin
+ // TODO print cache->mimeTypeInfo
+ }
+#endif
+
+ return plugin;
+}
+
+TQStringList KFileMetaInfoProvider::preferredKeys( const TQString& mimeType ) const
+{
+ KService::Ptr service =
+ KServiceTypeProfile::preferredService( mimeType, "KFilePlugin");
+
+ if ( !service || !service->isValid() )
+ {
+// kdDebug(7033) << "no valid service found\n";
+ return TQStringList();
+ }
+ return service->property("PreferredItems").toStringList();
+}
+
+TQStringList KFileMetaInfoProvider::preferredGroups( const TQString& mimeType ) const
+{
+ KService::Ptr service =
+ KServiceTypeProfile::preferredService( mimeType, "KFilePlugin");
+
+ if ( !service || !service->isValid() )
+ {
+// kdDebug(7033) << "no valid service found\n";
+ return TQStringList();
+ }
+ return service->property("PreferredGroups").toStringList();
+}
+
+const KFileMimeTypeInfo * KFileMetaInfoProvider::mimeTypeInfo( const TQString& mimeType )
+{
+ return mimeTypeInfo( mimeType, TQString::null );
+}
+
+const KFileMimeTypeInfo * KFileMetaInfoProvider::mimeTypeInfo( const TQString& mimeType, const TQString& protocol )
+{
+ //kdDebug(7033) << "mimeTypeInfo() : looking for plugin for protocol=" << protocol << " mimeType=" << mimeType << endl;
+ if ( !protocol.isEmpty() ) {
+ CachedPluginInfo *cache = m_plugins.find( protocol );
+ if ( cache && cache->mimeTypeInfo ) {
+ return cache->mimeTypeInfo;
+ }
+
+ if ( !cache ) {
+ loadAndRegisterPlugin( TQString::null, protocol );
+ cache = m_plugins.find( protocol );
+ if ( cache && cache->mimeTypeInfo ) {
+ return cache->mimeTypeInfo;
+ }
+ }
+ }
+
+ CachedPluginInfo *cache = m_plugins.find( mimeType );
+ if ( cache ) {
+ return cache->mimeTypeInfo;
+ }
+
+ loadAndRegisterPlugin( mimeType, TQString::null );
+ cache = m_plugins.find( mimeType );
+ if ( cache ) {
+ return cache->mimeTypeInfo;
+ }
+ return 0;
+}
+
+KFileMimeTypeInfo * KFileMetaInfoProvider::addMimeTypeInfo(
+ const TQString& mimeType )
+{
+
+ KFileMimeTypeInfo *info = m_pendingMimetypeInfos.find( mimeType );
+ Q_ASSERT( !info );
+ if ( !info )
+ {
+ info = new KFileMimeTypeInfo( mimeType );
+ m_pendingMimetypeInfos.insert( mimeType, info );
+ }
+
+ info->m_preferredKeys = preferredKeys( mimeType );
+ info->m_preferredGroups = preferredGroups( mimeType );
+
+ return info;
+}
+
+TQStringList KFileMetaInfoProvider::supportedMimeTypes() const
+{
+ TQStringList allMimeTypes;
+ TQString tdefilePlugin = "KFilePlugin";
+
+ TDETrader::OfferList offers = TDETrader::self()->query( "KFilePlugin" );
+ TDETrader::OfferListIterator it = offers.begin();
+ for ( ; it != offers.end(); ++it )
+ {
+ const TQStringList mimeTypes = (*it)->serviceTypes();
+ TQStringList::ConstIterator it2 = mimeTypes.begin();
+ for ( ; it2 != mimeTypes.end(); ++it2 )
+ if ( allMimeTypes.find( *it2 ) == allMimeTypes.end() &&
+ *it2 != tdefilePlugin ) // also in serviceTypes()
+ allMimeTypes.append( *it2 );
+ }
+
+ return allMimeTypes;
+}
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+
+// shared data of a KFileMetaInfoGroup
+class KFileMetaInfoGroup::Data : public TQShared
+{
+public:
+ Data(const TQString& _name)
+ : TQShared(),
+ name(_name),
+ mimeTypeInfo(0L),
+ dirty( false ),
+ added( false )
+ {}
+
+ // we use this one for the streaming operators
+ Data() : mimeTypeInfo(0L) {}
+ ~Data() {
+ if ( this == null )
+ delete mimeTypeInfo;
+ };
+
+ TQString name;
+ TQMap<TQString, KFileMetaInfoItem> items;
+ const KFileMimeTypeInfo* mimeTypeInfo;
+ TQStringList removedItems;
+ bool dirty :1;
+ bool added :1;
+
+ static Data* null;
+ static Data* makeNull();
+
+};
+
+KFileMetaInfoGroup::KFileMetaInfoGroup( const TQString& name,
+ const KFileMimeTypeInfo* info )
+ : d(new Data( name ) )
+{
+ d->mimeTypeInfo = info;
+}
+
+KFileMetaInfoGroup::KFileMetaInfoGroup( const KFileMetaInfoGroup& original )
+{
+ // operator= does everything that's necessary
+ d = Data::makeNull();
+ *this = original;
+}
+
+KFileMetaInfoGroup::KFileMetaInfoGroup()
+{
+ d = Data::makeNull();
+}
+
+KFileMetaInfoGroup::~KFileMetaInfoGroup()
+{
+ deref();
+}
+
+const KFileMetaInfoGroup& KFileMetaInfoGroup::operator= (const KFileMetaInfoGroup& info )
+{
+ if (d != info.d)
+ {
+ deref();
+ // first deref the old one
+ d = info.d;
+ // and now ref the new one
+ ref();
+ }
+ return *this;
+}
+
+bool KFileMetaInfoGroup::isValid() const
+{
+ // We don't call makeNull here since it isn't necassery, see deref()
+ return d != Data::null;
+}
+
+bool KFileMetaInfoGroup::isEmpty() const
+{
+ return d->items.isEmpty();
+}
+
+TQStringList KFileMetaInfoGroup::preferredKeys() const
+{
+ assert(isValid());
+ TQStringList list = keys();
+ TQStringList newlist;
+ TQStringList preferredKeys = d->mimeTypeInfo->preferredKeys();
+ TQStringList::Iterator pref;
+ TQStringList::Iterator begin = preferredKeys.begin();
+ TQStringList::Iterator end = preferredKeys.end();
+
+ // move all keys from the preferred keys that are in our list to a new list
+ for ( pref = begin; pref!=end; ++pref )
+ {
+ TQStringList::Iterator item = list.find(*pref);
+ if ( item != list.end() )
+ {
+ newlist.append( *item );
+ list.remove(item);
+ }
+ }
+
+ // now the old list only contains the non-preferred items, so we
+ // add the remaining ones to newlist
+ newlist += list;
+
+ return newlist;
+}
+
+TQStringList KFileMetaInfoGroup::keys() const
+{
+ if (d == Data::makeNull())
+ kdWarning(7033) << "attempt to get the keys of "
+ "an invalid metainfo group";
+
+ TQStringList list;
+
+ // make a TQStringList with all available keys
+ TQMapConstIterator<TQString, KFileMetaInfoItem> it;
+ for (it = d->items.begin(); it!=d->items.end(); ++it)
+ {
+ list.append(it.data().key());
+// kdDebug(7033) << "Item " << it.data().key() << endl;
+ }
+ return list;
+}
+
+TQString KFileMetaInfoGroup::translatedName() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->groupInfo(d->name)->translatedName();
+}
+
+TQStringList KFileMetaInfoGroup::supportedKeys() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->groupInfo(d->name)->supportedKeys();
+}
+
+bool KFileMetaInfoGroup::supportsVariableKeys() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->groupInfo(d->name)->supportsVariableKeys();
+}
+
+bool KFileMetaInfoGroup::contains( const TQString& key ) const
+{
+ return d->items.contains(key);
+}
+
+KFileMetaInfoItem KFileMetaInfoGroup::item( const TQString& key) const
+{
+ TQMapIterator<TQString,KFileMetaInfoItem> it = d->items.find( key );
+ if ( it != d->items.end() )
+ return it.data();
+
+ return KFileMetaInfoItem();
+}
+
+KFileMetaInfoItem KFileMetaInfoGroup::item(uint hint) const
+{
+ TQMapIterator<TQString, KFileMetaInfoItem> it;
+
+ for (it = d->items.begin(); it!=d->items.end(); ++it)
+ if (it.data().hint() == hint)
+ return it.data();
+
+ return KFileMetaInfoItem();
+}
+
+TQString KFileMetaInfoGroup::name() const
+{
+ return d->name;
+}
+
+uint KFileMetaInfoGroup::attributes() const
+{
+ assert(isValid());
+ return d->mimeTypeInfo->groupInfo(d->name)->attributes();
+}
+
+void KFileMetaInfoGroup::setAdded()
+{
+ d->added = true;
+}
+
+bool KFileMetaInfoGroup::isModified() const
+{
+ return d->dirty;
+}
+
+void KFileMetaInfoGroup::ref()
+{
+ if (d != Data::null) d->ref();
+
+}
+
+void KFileMetaInfoGroup::deref()
+{
+ // We don't call makeNull here since it isn't necassery:
+ // If d is equal to null it means that null is initialized already.
+ // null is 0L when it hasn't been initialized and d is never 0L.
+ if ((d != Data::null) && d->deref())
+ {
+// kdDebug(7033) << "metainfo group " << d->name
+// << " is finally deleted\n";
+ delete d;
+ d = 0;
+ }
+
+}
+
+KFileMetaInfoItem KFileMetaInfoGroup::addItem( const TQString& key )
+{
+ assert(isValid());
+ TQMapIterator<TQString,KFileMetaInfoItem> it = d->items.find( key );
+ if ( it != d->items.end() )
+ return it.data();
+
+ const KFileMimeTypeInfo::GroupInfo* ginfo = d->mimeTypeInfo->groupInfo(d->name);
+
+ if ( !ginfo ) {
+ Q_ASSERT( ginfo );
+ return KFileMetaInfoItem();
+ }
+
+ const KFileMimeTypeInfo::ItemInfo* info = ginfo->itemInfo(key);
+
+ if ( !info ) {
+ Q_ASSERT( info );
+ return KFileMetaInfoItem();
+ }
+
+ KFileMetaInfoItem item;
+
+ if (info->isVariableItem())
+ item = KFileMetaInfoItem(ginfo->variableItemInfo(), key, TQVariant());
+ else
+ item = KFileMetaInfoItem(info, key, TQVariant());
+
+ d->items.insert(key, item);
+ item.setAdded(); // mark as added
+ d->dirty = true; // mark ourself as dirty, too
+ return item;
+}
+
+bool KFileMetaInfoGroup::removeItem( const TQString& key )
+{
+ if (!isValid())
+ {
+ kdDebug(7033) << "trying to remove an item from an invalid group\n";
+ return false;
+ }
+
+ TQMapIterator<TQString, KFileMetaInfoItem> it = d->items.find(key);
+ if ( it==d->items.end() )
+ {
+ kdDebug(7033) << "trying to remove the non existant item " << key << "\n";
+ return false;
+ }
+
+ if (!((*it).attributes() & KFileMimeTypeInfo::Removable))
+ {
+ kdDebug(7033) << "trying to remove a non removable item\n";
+ return false;
+ }
+
+ (*it).setRemoved();
+ d->items.remove(it);
+ d->removedItems.append(key);
+ d->dirty = true;
+ return true;
+}
+
+TQStringList KFileMetaInfoGroup::removedItems()
+{
+ return d->removedItems;
+}
+
+KFileMetaInfoItem KFileMetaInfoGroup::appendItem(const TQString& key,
+ const TQVariant& value)
+{
+ //KDE4 enforce (value.type() == d->mimeTypeInfo->type())
+ assert(isValid());
+ const KFileMimeTypeInfo::GroupInfo* ginfo = d->mimeTypeInfo->groupInfo(d->name);
+ if ( !ginfo ) {
+ kdWarning() << "Trying to append a Metadata item for a non-existant group:" << d->name << endl;
+ return KFileMetaInfoItem();
+ }
+ const KFileMimeTypeInfo::ItemInfo* info = ginfo->itemInfo(key);
+ if ( !info ) {
+ kdWarning() << "Trying to append a Metadata item for an unknown key (no ItemInfo): " << key << endl;
+ return KFileMetaInfoItem();
+ }
+
+ KFileMetaInfoItem item;
+
+ if (info->key().isNull())
+ item = KFileMetaInfoItem(ginfo->variableItemInfo(), key, value);
+ else
+ item = KFileMetaInfoItem(info, key, value);
+
+ kdDebug(7033) << "KFileMetaInfogroup inserting a " << key << endl;
+
+ d->items.insert(key, item);
+ return item;
+}
+
+KFileMetaInfoGroup::Data* KFileMetaInfoGroup::Data::null = 0L;
+static KStaticDeleter<KFileMetaInfoGroup::Data> sd_KFileMetaInfoGroupData;
+
+KFileMetaInfoGroup::Data* KFileMetaInfoGroup::Data::makeNull()
+{
+ if (!null)
+ {
+ // We deliberately do not reset "null" after it has been destroyed!
+ // Otherwise we will run into problems later in ~KFileMetaInfoItem
+ // where the d-pointer is compared against null.
+ null = new Data(TQString::null);
+ null->mimeTypeInfo = new KFileMimeTypeInfo();
+ sd_KFileMetaInfoGroupData.setObject( null );
+ }
+ return null;
+}
+
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+KFileMimeTypeInfo::KFileMimeTypeInfo( const TQString& mimeType )
+ : m_mimeType( mimeType )
+{
+ m_groups.setAutoDelete( true );
+}
+
+KFileMimeTypeInfo::~KFileMimeTypeInfo()
+{
+}
+
+const KFileMimeTypeInfo::GroupInfo * KFileMimeTypeInfo::groupInfo( const TQString& group ) const
+{
+ return m_groups.find( group );
+}
+
+KFileMimeTypeInfo::GroupInfo * KFileMimeTypeInfo::addGroupInfo(
+ const TQString& name, const TQString& translatedName )
+{
+ GroupInfo* group = new GroupInfo( name, translatedName );
+ m_groups.insert(name, group);
+ return group;
+}
+
+TQStringList KFileMimeTypeInfo::supportedGroups() const
+{
+ TQStringList list;
+ TQDictIterator<GroupInfo> it( m_groups );
+ for ( ; it.current(); ++it )
+ list.append( it.current()->name() );
+
+ return list;
+}
+
+TQStringList KFileMimeTypeInfo::translatedGroups() const
+{
+ TQStringList list;
+ TQDictIterator<GroupInfo> it( m_groups );
+ for ( ; it.current(); ++it )
+ list.append( it.current()->translatedName() );
+
+ return list;
+}
+
+TQStringList KFileMimeTypeInfo::supportedKeys() const
+{
+ // not really efficient, but not those are not large lists, probably.
+ // maybe cache the result?
+ TQStringList keys;
+ TQStringList::ConstIterator lit;
+ TQDictIterator<GroupInfo> it( m_groups );
+ for ( ; it.current(); ++it ) { // need to nuke dupes
+ TQStringList list = it.current()->supportedKeys();
+ for ( lit = list.begin(); lit != list.end(); ++lit ) {
+ if ( keys.find( *lit ) == keys.end() )
+ keys.append( *lit );
+ }
+ }
+
+ return keys;
+}
+
+TQValidator * KFileMimeTypeInfo::createValidator(const TQString& group,
+ const TQString& key,
+ TQObject *parent,
+ const char *name) const
+{
+ KFilePlugin* plugin = KFileMetaInfoProvider::self()->plugin(m_mimeType);
+ if (plugin) return plugin->createValidator(mimeType(), group, key,
+ parent, name);
+ return 0;
+}
+
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+KFileMimeTypeInfo::GroupInfo::GroupInfo( const TQString& name,
+ const TQString& translatedName )
+ : m_name( name ),
+ m_translatedName( translatedName ),
+ m_attr( 0 ),
+ m_variableItemInfo( 0 )
+
+{
+ m_itemDict.setAutoDelete( true );
+}
+
+KFileMimeTypeInfo::GroupInfo::~GroupInfo()
+{
+ delete m_variableItemInfo;
+}
+
+const KFileMimeTypeInfo::ItemInfo * KFileMimeTypeInfo::GroupInfo::itemInfo( const TQString& key ) const
+{
+ ItemInfo* item = m_itemDict.find( key );
+
+ // if we the item isn't found and variable keys are supported, we need to
+ // return the default variable key iteminfo.
+ if (!item && m_variableItemInfo)
+ {
+ return m_variableItemInfo;
+ }
+ return item;
+}
+
+KFileMimeTypeInfo::ItemInfo* KFileMimeTypeInfo::GroupInfo::addItemInfo(
+ const TQString& key, const TQString& translatedKey,
+ TQVariant::Type type)
+{
+// kdDebug(7034) << key << "(" << translatedKey << ") -> " << TQVariant::typeToName(type) << endl;
+
+ ItemInfo* item = new ItemInfo(key, translatedKey, type);
+ m_supportedKeys.append(key);
+ m_itemDict.insert(key, item);
+ return item;
+}
+
+
+void KFileMimeTypeInfo::GroupInfo::addVariableInfo( TQVariant::Type type,
+ uint attr )
+{
+ // just make sure that it's not already there
+ delete m_variableItemInfo;
+ m_variableItemInfo = new ItemInfo(TQString::null, TQString::null, type);
+ m_variableItemInfo->m_attr = attr;
+}
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+TQString KFileMimeTypeInfo::ItemInfo::string(const TQVariant& value, bool mangle) const
+{
+ TQString s;
+
+ switch (value.type())
+ {
+ case TQVariant::Invalid :
+ return "---";
+
+ case TQVariant::Bool :
+ s = value.toBool() ? i18n("Yes") : i18n("No");
+ break;
+
+ case TQVariant::Int :
+ if (unit() == KFileMimeTypeInfo::Seconds)
+ {
+ int seconds = value.toInt() % 60;
+ int minutes = value.toInt() / 60 % 60;
+ int hours = value.toInt() / 3600;
+ s = hours ? TQString().sprintf("%d:%02d:%02d",hours, minutes, seconds)
+ : TQString().sprintf("%02d:%02d", minutes, seconds);
+ return s; // no suffix wanted
+ }
+ else if (unit() == KFileMimeTypeInfo::Bytes)
+ {
+ // convertSize already adds the correct suffix
+ return TDEIO::convertSize(value.toInt());
+ }
+ else if (unit() == KFileMimeTypeInfo::KiloBytes)
+ {
+ // convertSizeFromKB already adds the correct suffix
+ return TDEIO::convertSizeFromKB(value.toInt());
+ }
+ else
+ s = TDEGlobal::locale()->formatNumber( value.toInt() , 0);
+ break;
+
+ case TQVariant::LongLong :
+ s = TDEGlobal::locale()->formatNumber( value.toLongLong(), 0 );
+ break;
+
+ case TQVariant::ULongLong :
+ if ( unit() == KFileMimeTypeInfo::Bytes )
+ return TDEIO::convertSize( value.toULongLong() );
+ else if ( unit() == KFileMimeTypeInfo::KiloBytes )
+ return TDEIO::convertSizeFromKB( value.toULongLong() );
+ else
+ s = TDEGlobal::locale()->formatNumber( value.toULongLong(), 0 );
+ break;
+
+ case TQVariant::UInt :
+ s = TDEGlobal::locale()->formatNumber( value.toUInt() , 0);
+ break;
+
+ case TQVariant::Double :
+ s = TDEGlobal::locale()->formatNumber( value.toDouble(), 3);
+ break;
+
+ case TQVariant::Date :
+ s = TDEGlobal::locale()->formatDate( value.toDate(), true );
+ break;
+
+ case TQVariant::Time :
+ s = TDEGlobal::locale()->formatTime( value.toTime(), true );
+ break;
+
+ case TQVariant::DateTime :
+ s = TDEGlobal::locale()->formatDateTime( value.toDateTime(),
+ true, true );
+ break;
+
+ case TQVariant::Size :
+ s = TQString("%1 x %2").arg(value.toSize().width())
+ .arg(value.toSize().height());
+ break;
+
+ case TQVariant::Point :
+ s = TQString("%1/%2").arg(value.toSize().width())
+ .arg(value.toSize().height());
+ break;
+
+ default:
+ s = value.toString();
+ }
+
+ if (mangle && !s.isNull())
+ {
+ s.prepend(prefix());
+ s.append(" " + suffix());
+ }
+ return s;
+}
+
+
+///////////////////////////////////////////////////////////////////
+///////////////////////////////////////////////////////////////////
+
+
+
+// stream operators
+
+/* serialization of a KFileMetaInfoItem:
+ first a bool that says if the items is valid, and if yes,
+ all the elements of the Data
+*/
+TDEIO_EXPORT TQDataStream& operator <<(TQDataStream& s, const KFileMetaInfoItem& item )
+{
+
+ KFileMetaInfoItem::Data* d = item.d;
+
+ // if the object is invalid, put only a char in the stream
+ bool isValid = item.isValid();
+ s << isValid;
+ // ### what do about mimetypeInfo ?
+ if (isValid)
+ s << d->key
+ << d->value
+ << d->dirty
+ << d->added
+ << d->removed;
+
+ return s;
+}
+
+
+TDEIO_EXPORT TQDataStream& operator >>(TQDataStream& s, KFileMetaInfoItem& item )
+{
+ bool isValid;
+ s >> isValid;
+
+ if (!isValid)
+ {
+ item = KFileMetaInfoItem();
+ return s;
+ }
+
+ // we need a new object for our data
+ item.deref();
+ item.d = new KFileMetaInfoItem::Data();
+
+ // ### what do about mimetypeInfo ?
+ bool dirty, added, removed;
+ s >> item.d->key
+ >> item.d->value
+ >> dirty
+ >> added
+ >> removed;
+ item.d->dirty = dirty;
+ item.d->added = added;
+ item.d->removed = removed;
+
+ return s;
+}
+
+
+// serialization of a KFileMetaInfoGroup
+// we serialize the name of the mimetype here instead of the mimetype info
+// on the other side, we can simply use this to ask the provider for the info
+TDEIO_EXPORT TQDataStream& operator <<(TQDataStream& s, const KFileMetaInfoGroup& group )
+{
+ KFileMetaInfoGroup::Data* d = group.d;
+
+ // if the object is invalid, put only a byte in the stream
+ bool isValid = group.isValid();
+
+ s << isValid;
+ if (isValid)
+ {
+ s << d->name
+ << d->items
+ << d->mimeTypeInfo->mimeType();
+ }
+ return s;
+}
+
+TDEIO_EXPORT TQDataStream& operator >>(TQDataStream& s, KFileMetaInfoGroup& group )
+{
+ TQString mimeType;
+ bool isValid;
+ s >> isValid;
+
+ // if it's invalid, there is not much to do
+ if (!isValid)
+ {
+ group = KFileMetaInfoGroup();
+ return s;
+ }
+
+ // we need a new object for our data
+ group.deref();
+ group.d = new KFileMetaInfoGroup::Data();
+
+ s >> group.d->name
+ >> group.d->items
+ >> mimeType;
+
+ group.d->mimeTypeInfo = KFileMetaInfoProvider::self()->mimeTypeInfo(mimeType);
+
+ // we need to set the item info for the items here
+ TQMapIterator<TQString, KFileMetaInfoItem> it = group.d->items.begin();
+ for ( ; it != group.d->items.end(); ++it)
+ {
+ (*it).d->mimeTypeInfo = group.d->mimeTypeInfo->groupInfo(group.d->name)
+ ->itemInfo((*it).key());
+ }
+
+ return s;
+}
+
+// serialization of a KFileMetaInfo object
+// we serialize the name of the mimetype here instead of the mimetype info
+// on the other side, we can simply use this to ask the provider for the info
+TDEIO_EXPORT TQDataStream& operator <<(TQDataStream& s, const KFileMetaInfo& info )
+{
+ KFileMetaInfo::Data* d = info.d;
+
+ // if the object is invalid, put only a byte that tells this
+ bool isValid = info.isValid();
+
+ s << isValid;
+ if (isValid)
+ {
+ s << d->url
+ << d->what
+ << d->groups
+ << d->mimeTypeInfo->mimeType();
+ }
+ return s;
+}
+
+TDEIO_EXPORT TQDataStream& operator >>(TQDataStream& s, KFileMetaInfo& info )
+{
+ TQString mimeType;
+ bool isValid;
+ s >> isValid;
+
+ // if it's invalid, there is not much to do
+ if (!isValid)
+ {
+ info = KFileMetaInfo();
+ return s;
+ }
+
+ // we need a new object for our data
+ info.deref();
+ info.d = new KFileMetaInfo::Data();
+
+ s >> info.d->url
+ >> info.d->what
+ >> info.d->groups
+ >> mimeType;
+ info.d->mimeTypeInfo = KFileMetaInfoProvider::self()->mimeTypeInfo(mimeType);
+
+ return s;
+}
+
+#include "tdefilemetainfo.moc"