summaryrefslogtreecommitdiffstats
path: root/kexi/core/kexiblobbuffer.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kexi/core/kexiblobbuffer.cpp')
-rw-r--r--kexi/core/kexiblobbuffer.cpp373
1 files changed, 373 insertions, 0 deletions
diff --git a/kexi/core/kexiblobbuffer.cpp b/kexi/core/kexiblobbuffer.cpp
new file mode 100644
index 00000000..d66d69af
--- /dev/null
+++ b/kexi/core/kexiblobbuffer.cpp
@@ -0,0 +1,373 @@
+/* This file is part of the KDE project
+ Copyright (C) 2005 Jaroslaw Staniek <js@iidea.pl>
+
+ 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 "kexiblobbuffer.h"
+
+#include <assert.h>
+
+#include <qfile.h>
+#include <qfileinfo.h>
+#include <qbuffer.h>
+
+#include <kdebug.h>
+#include <kstaticdeleter.h>
+#include <kimageio.h>
+
+#include <kexidb/connection.h>
+
+static KStaticDeleter<KexiBLOBBuffer> m_bufferDeleter;
+static KexiBLOBBuffer* m_buffer = 0;
+
+//-----------------
+
+class KexiBLOBBuffer::Private
+{
+ public:
+ Private()
+ : maxId(0)
+ , inMemoryItems(1009)
+ , storedItems(1009)
+ , itemsByURL(1009)
+ {
+ }
+ Id_t maxId; //!< Used to compute maximal recently used identifier for unstored BLOB
+//! @todo will be changed to QHash<quint64, Item>
+ QIntDict<Item> inMemoryItems; //!< for unstored BLOBs
+ QIntDict<Item> storedItems; //!< for stored items
+ QDict<Item> itemsByURL;
+ QGuardedPtr<KexiDB::Connection> conn;
+};
+
+//-----------------
+
+KexiBLOBBuffer::Handle::Handle(Item* item)
+ : m_item(item)
+{
+ if (m_item)
+ m_item->refs++;
+}
+
+KexiBLOBBuffer::Handle::Handle(const Handle& handle)
+{
+ *this = handle;
+}
+
+KexiBLOBBuffer::Handle::Handle()
+ : m_item(0)
+{
+}
+
+KexiBLOBBuffer::Handle::~Handle()
+{
+ if (m_item) {
+ m_item->refs--;
+ if (m_item->refs<=0)
+ KexiBLOBBuffer::self()->removeItem(m_item->id, m_item->stored);
+ }
+}
+
+KexiBLOBBuffer::Handle& KexiBLOBBuffer::Handle::operator=(const Handle& handle)
+{
+ m_item = handle.m_item;
+ if (m_item)
+ m_item->refs++;
+ return *this;
+}
+
+void KexiBLOBBuffer::Handle::setStoredWidthID(KexiBLOBBuffer::Id_t id)
+{
+ if (!m_item)
+ return;
+ if (m_item->stored) {
+ kdWarning() << "KexiBLOBBuffer::Handle::setStoredWidthID(): object for id=" << id
+ << " is aleady stored" << endl;
+ return;
+ }
+
+ KexiBLOBBuffer::self()->takeItem(m_item);
+ m_item->id = id; //new id
+ m_item->stored = true;
+//! @todo What about other handles for this item?
+//! @todo They were assuming it's unstored item, but it's stored now....
+ KexiBLOBBuffer::self()->insertItem(m_item);
+}
+
+//-----------------
+
+KexiBLOBBuffer::Item::Item(const QByteArray& data, KexiBLOBBuffer::Id_t ident, bool _stored,
+ const QString& _name, const QString& _caption, const QString& _mimeType,
+ Id_t _folderId, const QPixmap& pixmap)
+ : name(_name), caption(_caption), mimeType(_mimeType), refs(0),
+ id(ident), folderId(_folderId), stored(_stored),
+ m_pixmapLoaded(new bool(false)/*workaround for pixmap() const*/)
+{
+ if (pixmap.isNull())
+ m_pixmap = new QPixmap();
+ else
+ m_pixmap = new QPixmap(pixmap);
+
+ if (data.isEmpty())
+ m_data = new QByteArray();
+ else
+ m_data = new QByteArray(data);
+}
+
+KexiBLOBBuffer::Item::~Item()
+{
+ kexipluginsdbg << "KexiBLOBBuffer::Item::~Item()" << endl;
+ delete m_pixmap;
+ m_pixmap = 0;
+ delete m_data;
+ m_data = 0;
+ delete m_pixmapLoaded;
+}
+
+QPixmap KexiBLOBBuffer::Item::pixmap() const
+{
+ if (!*m_pixmapLoaded && m_pixmap->isNull() && !m_data->isEmpty()) {
+ QString type( KImageIO::typeForMime(mimeType) );
+ if (!KImageIO::canRead( type ) || !m_pixmap->loadFromData(*m_data, type.latin1())) {
+ //! @todo inform about error?
+ }
+ *m_pixmapLoaded = true;
+ }
+ return *m_pixmap;
+}
+
+QByteArray KexiBLOBBuffer::Item::data() const
+{
+ if (!m_data->isEmpty())
+ return *m_data;
+
+ if (m_data->isEmpty() && m_pixmap->isNull())
+ return QByteArray();
+
+ if (m_data->isEmpty() && !m_pixmap->isNull()) {
+ //convert pixmap to byte array
+ //(do it only on demand)
+ QBuffer buffer( *m_data );
+ buffer.open( IO_WriteOnly );
+ m_pixmap->save( &buffer, mimeType.isEmpty() ? (const char*)"PNG"/*! @todo default? */ : mimeType.latin1() );
+ }
+ return *m_data;
+}
+
+//-----------------
+
+KexiBLOBBuffer::KexiBLOBBuffer()
+ : QObject()
+ , d(new Private())
+{
+ Q_ASSERT(!m_buffer);
+ d->inMemoryItems.setAutoDelete(true);
+ d->storedItems.setAutoDelete(true);
+}
+
+KexiBLOBBuffer::~KexiBLOBBuffer()
+{
+ delete d;
+}
+
+KexiBLOBBuffer::Handle KexiBLOBBuffer::insertPixmap(const KURL& url)
+{
+ if (url.isEmpty() )
+ return KexiBLOBBuffer::Handle();
+ if (!url.isValid()) {
+ kexipluginswarn << "::insertPixmap: INVALID URL '" << url << "'" << endl;
+ return KexiBLOBBuffer::Handle();
+ }
+//! @todo what about searching by filename only and then compare data?
+ Item * item = d->itemsByURL.find(url.prettyURL());
+ if (item)
+ return KexiBLOBBuffer::Handle(item);
+
+ QString fileName = url.isLocalFile() ? url.path() : url.prettyURL();
+//! @todo download the file if remote, then set fileName properly
+ QFile f(fileName);
+ if (!f.open(IO_ReadOnly)) {
+ //! @todo err msg
+ return KexiBLOBBuffer::Handle();
+ }
+ QString mimeType( KImageIO::mimeType( fileName ) );
+
+ QByteArray data( f.readAll() );
+ if (f.status()!=IO_Ok) {
+ //! @todo err msg
+ return KexiBLOBBuffer::Handle();
+ }
+ QFileInfo fi(url.fileName());
+ QString caption(fi.baseName().replace('_', " ").simplifyWhiteSpace());
+
+ item = new Item(data, ++d->maxId, /*!stored*/false, url.fileName(), caption, mimeType);
+ insertItem(item);
+
+ //cache
+ item->prettyURL = url.prettyURL();
+ d->itemsByURL.replace(url.prettyURL(), item);
+ return KexiBLOBBuffer::Handle(item);
+}
+
+KexiBLOBBuffer::Handle KexiBLOBBuffer::insertObject(const QByteArray& data,
+ const QString& name, const QString& caption, const QString& mimeType, KexiBLOBBuffer::Id_t identifier)
+{
+ KexiBLOBBuffer::Id_t newIdentifier;
+ if (identifier>0)
+ newIdentifier = identifier;
+ else
+ newIdentifier = ++d->maxId;
+
+ Item *item = new Item(data, newIdentifier, identifier>0, name, caption, mimeType);
+ insertItem( item );
+ return KexiBLOBBuffer::Handle(item);
+}
+
+KexiBLOBBuffer::Handle KexiBLOBBuffer::insertPixmap(const QPixmap& pixmap)
+{
+ if (pixmap.isNull())
+ return KexiBLOBBuffer::Handle();
+
+ Item * item = new Item(
+ QByteArray(), //(pixmap will be converted to byte array on demand)
+ ++d->maxId,
+ false, //not stored
+ QString::null,
+ QString::null,
+ "image/png", //!< @todo OK? What about jpegs?
+ 0, //folder id
+ pixmap);
+
+ insertItem(item);
+ return KexiBLOBBuffer::Handle(item);
+}
+
+KexiBLOBBuffer::Handle KexiBLOBBuffer::objectForId(Id_t id, bool stored)
+{
+ if (id<=0)
+ return KexiBLOBBuffer::Handle();
+ if (stored) {
+ Item *item = d->storedItems.find(id);
+ if (item || !d->conn)
+ return KexiBLOBBuffer::Handle(item);
+ //retrieve stored BLOB:
+
+//#if 0
+ assert(d->conn);
+ KexiDB::TableSchema *blobsTable = d->conn->tableSchema("kexi__blobs");
+ if (!blobsTable) {
+ //! @todo err msg
+ return KexiBLOBBuffer::Handle();
+ }
+/* QStringList where;
+ where << "o_id";
+ KexiDB::PreparedStatement::Ptr st = d->conn->prepareStatement(
+ KexiDB::PreparedStatement::SelectStatement, *blobsTable, where);*/
+//! @todo use PreparedStatement
+ KexiDB::QuerySchema schema;
+ schema.addField( blobsTable->field("o_data") );
+ schema.addField( blobsTable->field("o_name") );
+ schema.addField( blobsTable->field("o_caption") );
+ schema.addField( blobsTable->field("o_mime") );
+ schema.addField( blobsTable->field("o_folder_id") );
+ schema.addToWhereExpression(blobsTable->field("o_id"), QVariant((Q_LLONG)id));
+
+ KexiDB::RowData rowData;
+ tristate res = d->conn->querySingleRecord(
+ schema,
+// QString::fromLatin1("SELECT o_data, o_name, o_caption, o_mime FROM kexi__blobs where o_id=")
+// +QString::number(id),
+ rowData);
+ if (res!=true || rowData.size()<4) {
+ //! @todo err msg
+ kdWarning() << "KexiBLOBBuffer::objectForId("<<id<<","<<stored
+ <<"): res!=true || rowData.size()<4; res=="<<res.toString()<<" rowData.size()=="<<rowData.size()<< endl;
+ return KexiBLOBBuffer::Handle();
+ }
+
+ item = new Item(
+ rowData[0].toByteArray(),
+ id,
+ true, //stored
+ rowData[1].toString(),
+ rowData[2].toString(),
+ rowData[3].toString(),
+ (Id_t)rowData[4].toInt() //!< @todo folder id: fix Id_t for Qt4
+ );
+
+ insertItem(item);
+ return KexiBLOBBuffer::Handle(item);
+//#endif
+ }
+ else
+ return KexiBLOBBuffer::Handle(d->inMemoryItems.find(id));
+}
+
+KexiBLOBBuffer::Handle KexiBLOBBuffer::objectForId(Id_t id)
+{
+ KexiBLOBBuffer::Handle h(objectForId(id, false/*!stored*/));
+ if (h)
+ return h;
+ return objectForId(id, true/*stored*/);
+}
+
+void KexiBLOBBuffer::removeItem(Id_t id, bool stored)
+{
+ Item *item;
+ if (stored)
+ item = d->storedItems.take(id);
+ else
+ item = d->inMemoryItems.take(id);
+
+ if (item && !item->prettyURL.isEmpty()) {
+ d->itemsByURL.remove(item->prettyURL);
+ }
+ delete item;
+}
+
+void KexiBLOBBuffer::takeItem(Item *item)
+{
+ assert(item);
+ if (item->stored)
+ d->storedItems.take(item->id);
+ else
+ d->inMemoryItems.take(item->id);
+}
+
+void KexiBLOBBuffer::insertItem(Item *item)
+{
+ assert(item);
+ if (item->stored)
+ d->storedItems.insert(item->id, item);
+ else
+ d->inMemoryItems.insert(item->id, item);
+}
+
+void KexiBLOBBuffer::setConnection(KexiDB::Connection *conn)
+{
+ KexiBLOBBuffer::self()->d->conn = conn;
+}
+
+KexiBLOBBuffer* KexiBLOBBuffer::self()
+{
+ if(!m_buffer) {
+ m_bufferDeleter.setObject( m_buffer, new KexiBLOBBuffer() );
+ }
+ return m_buffer;
+}
+
+#include "kexiblobbuffer.moc"