kdelibs/kde3support/kdecore/k3urldrag.cpp
2014-11-13 01:04:59 +02:00

284 lines
8.6 KiB
C++

/* This file is part of the KDE project
Copyright (C) 2000 David Faure <faure@kde.org>
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.
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 "k3urldrag.h"
#include <Qt3Support/Q3CString>
#include <Qt3Support/Q3StrIList>
#include <Qt3Support/Q3ColorDrag>
#include <QtGui/QFont>
#include <unistd.h>
#include <kglobal.h>
#include <klocale.h>
#include <kdebug.h>
class K3URLDragPrivate
{
public:
bool m_exportAsText;
};
K3URLDrag::K3URLDrag( const KUrl::List &urls, QWidget* dragSource )
: Q3UriDrag(dragSource), m_metaData(), d( 0 )
{
init(urls);
}
K3URLDrag::K3URLDrag( const KUrl::List &urls,
const QMap<QString,QString>& metaData,
QWidget* dragSource )
: Q3UriDrag(dragSource), m_metaData(metaData), d( 0 )
{
init(urls);
}
K3URLDrag::~K3URLDrag()
{
delete d;
}
void K3URLDrag::init(const KUrl::List &urls)
{
KUrl::List::ConstIterator uit = urls.begin();
KUrl::List::ConstIterator uEnd = urls.end();
// Get each URL encoded in utf8 - and since we get it in escaped
// form on top of that, .toLatin1().constData() is fine.
for ( ; uit != uEnd ; ++uit )
{
m_urls.append( urlToString(*uit).toLatin1().constData() );
}
setUris(m_urls);
}
void K3URLDrag::setExportAsText( bool exp )
{
// For now d is only used here, so create it on demand
if ( !d )
d = new K3URLDragPrivate;
d->m_exportAsText = exp;
}
K3URLDrag * K3URLDrag::newDrag( const KUrl::List &urls, QWidget* dragSource )
{
return new K3URLDrag( urls, QMap<QString, QString>(), dragSource );
}
K3URLDrag * K3URLDrag::newDrag( const KUrl::List &urls, const QMap<QString, QString>& metaData,
QWidget* dragSource )
{
return new K3URLDrag( urls, metaData, dragSource );
}
QMap<QString, QString> &K3URLDrag::metaData()
{
return m_metaData;
}
bool K3URLDrag::decode( const QMimeSource *e, KUrl::List &uris )
{
// x-kde4-urilist is the same format as text/uri-list, but contains
// KDE-aware urls, like media:/ and system:/, whereas text/uri-list is resolved to
// local files.
if ( e->provides( "application/x-kde4-urilist" ) ) {
QByteArray payload = e->encodedData( "application/x-kde4-urilist" );
if ( payload.size() ) {
int c=0;
const char* d = payload.data();
while (c < payload.size() && d[c]) {
int f = c;
// Find line end
while (c < payload.size() && d[c] && d[c]!='\r'
&& d[c] != '\n')
c++;
Q3CString s(d+f,c-f+1);
if ( s[0] != '#' ) // non-comment?
uris.append(stringToUrl(s));
// Skip junk
while (c < payload.size() && d[c] &&
(d[c]=='\n' || d[c]=='\r'))
c++;
}
return !uris.isEmpty();
}
}
Q3StrList lst;
Q3UriDrag::decode( e, lst );
for (Q3StrListIterator it(lst); *it; ++it)
{
KUrl url = stringToUrl( *it );
if ( !url.isValid() )
{
uris.clear();
break;
}
uris.append( url );
}
return !uris.isEmpty();
}
bool K3URLDrag::decode( const QMimeSource *e, KUrl::List &uris, QMap<QString,QString>& metaData )
{
if ( decode( e, uris ) ) // first decode the URLs (see above)
{
QByteArray ba = e->encodedData( "application/x-kio-metadata" );
if ( ba.size() )
{
QString s = ba.data();
const QStringList l = s.split( "$@@$", QString::SkipEmptyParts );
QStringList::ConstIterator it = l.begin();
bool readingKey = true; // true, then false, then true, etc.
QString key;
for ( ; it != l.end(); ++it ) {
if ( readingKey )
key = *it;
else
metaData.replace( key, *it );
readingKey = !readingKey;
}
Q_ASSERT( readingKey ); // an odd number of items would be, well, odd ;-)
}
return true; // Success, even if no metadata was found
}
return false; // Couldn't decode the URLs
}
////
const char * K3URLDrag::format( int i ) const
{
if ( i == 0 )
return "text/uri-list";
else if ( i == 1 )
return "application/x-kio-metadata";
if ( d && d->m_exportAsText == false )
return 0;
if ( i == 2 )
return "text/plain";
else if ( i == 3 ) //Support this for apps that use plain XA_STRING clipboard
return "text/plain;charset=ISO-8859-1";
else if ( i == 4 ) //Support this for apps that use the UTF_STRING clipboard
return "text/plain;charset=UTF-8";
else return 0;
}
QByteArray K3URLDrag::encodedData( const char* mime ) const
{
QByteArray a;
QByteArray mimetype( mime );
if ( mimetype == "text/uri-list" )
return Q3UriDrag::encodedData( mime );
else if ( mimetype == "text/plain" )
{
QStringList uris;
for (Q3StrListIterator it(m_urls); *it; ++it)
uris.append(stringToUrl(*it).prettyUrl());
QByteArray s = uris.join( "\n" ).toLocal8Bit();
if( uris.count() > 1 ) // terminate last line, unless it's the only line
s.append( "\n" );
a.resize( s.length());
memcpy( a.data(), s.data(), s.length()); // no trailing zero in clipboard text
}
else if ( mimetype.toLower() == "text/plain;charset=iso-8859-1")
{
QStringList uris;
for (Q3StrListIterator it(m_urls); *it; ++it)
uris.append(stringToUrl(*it).url()); // was using ",4" - the mib for latin1
QByteArray s = uris.join( "\n" ).toLatin1();
if( uris.count() > 1 )
s.append( "\n" );
a.resize( s.length());
memcpy( a.data(), s.data(), s.length());
}
else if ( mimetype.toLower() == "text/plain;charset=utf-8")
{
QStringList uris;
for (Q3StrListIterator it(m_urls); *it; ++it)
uris.append(stringToUrl(*it).prettyUrl());
QByteArray s = uris.join( "\n" ).toUtf8();
if( uris.count() > 1 )
s.append( "\n" );
a.resize( s.length());
memcpy( a.data(), s.data(), s.length());
}
else if ( mimetype == "application/x-kio-metadata" )
{
if ( !m_metaData.isEmpty() )
{
QString s;
QMap<QString,QString>::ConstIterator it;
for( it = m_metaData.begin(); it != m_metaData.end(); ++it )
{
s += it.key();
s += "$@@$";
s += it.data();
s += "$@@$";
}
a.resize( s.length() + 1 );
memcpy( a.data(), s.toLatin1().constData(), a.size() );
}
}
return a;
}
KUrl K3URLDrag::stringToUrl(const QByteArray &s)
{
if (strncmp(s.data(), "file:", 5) == 0)
return KUrl(s /*, KGlobal::locale()->fileEncodingMib()*/);
return KUrl(s /*, 106*/); // 106 is mib enum for utf8 codec;
}
QString K3URLDrag::urlToString(const KUrl &url)
{
if (url.isLocalFile())
{
#if 1
return url.url(/*0 , KGlobal::locale()->fileEncodingMib()*/);
#else
// According to the XDND spec, file:/ URLs for DND must have
// the hostname part. But in really it just breaks many apps,
// so it's disabled for now.
QString s = url.url(0, KGlobal::locale()->fileEncodingMib());
if( !s.startsWith( "file://" ))
{
char hostname[257];
if ( gethostname( hostname, 255 ) == 0 )
{
hostname[256] = '\0';
return QString( "file://" ) + hostname + s.mid( 5 );
}
}
#endif
}
if ( url.protocol() == "mailto" ) {
return url.path();
}
return url.url(/*0 , 106*/); // 106 is mib enum for utf8 codec
}
// deprecated ctor
K3URLDrag::K3URLDrag( const Q3StrList & urls, const QMap<QString,QString>& metaData,
QWidget * dragSource ) :
Q3UriDrag( urls, dragSource ), m_urls( urls ), m_metaData( metaData ), d( 0 ) {}