You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
koffice/kexi/widget/tableview/kexiblobtableedit.cpp

596 lines
14 KiB

/* This file is part of the KDE project
Copyright (C) 2002 Peter Simonsson <psn@linux.se>
Copyright (C) 2004, 2006 Jaroslaw Staniek <js@iidea.pl>
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 "kexiblobtableedit.h"
#include <stdlib.h>
#include <tqdatastream.h>
#include <tqfile.h>
#include <tqpopupmenu.h>
#include <tqtextedit.h>
#include <tqlayout.h>
#include <tqstatusbar.h>
#include <tqlabel.h>
#include <tqpixmap.h>
#include <tqimage.h>
#include <tqpainter.h>
#include <tqtooltip.h>
#include <tqapplication.h>
#include <tqclipboard.h>
#include <tqbuffer.h>
#include <kdebug.h>
#include <ktempfile.h>
#include <kmimetype.h>
#include <kmimemagic.h>
#include <kuserprofile.h>
#include <kservice.h>
#include <kprocess.h>
#include <kopenwith.h>
#include <kurl.h>
#include <karrowbutton.h>
#include <klocale.h>
#include <kfiledialog.h>
#include <kio/job.h>
#include <kglobal.h>
#include <kiconloader.h>
#include <kpopupmenu.h>
#include <kstdaccel.h>
#include <kexiutils/utils.h>
#include <widget/utils/kexidropdownbutton.h>
#include <widget/utils/kexicontextmenuutils.h>
//! @internal
class KexiBlobTableEdit::Private
{
public:
Private()
: popup(0)
, readOnly(false)
, setValueInternalEnabled(true)
{
}
TQByteArray value;
KexiDropDownButton *button;
TQSize totalSize;
KexiImageContextMenu *popup;
bool readOnly : 1; //!< cached for slotUpdateActionsAvailabilityRequested()
bool setValueInternalEnabled : 1; //!< used to disable KexiBlobTableEdit::setValueInternal()
};
//======================================================
KexiBlobTableEdit::KexiBlobTableEdit(KexiTableViewColumn &column, TQWidget *parent)
: KexiTableEdit(column, parent)
, d ( new Private() )
{
setName("KexiBlobTableEdit");
// m_proc = 0;
// m_content = 0;
m_hasFocusableWidget = false;
d->button = new KexiDropDownButton( parentWidget() /*usually a viewport*/ );
d->button->hide();
TQToolTip::add(d->button, i18n("Click to show available actions for this cell"));
d->popup = new KexiImageContextMenu(this);
d->popup->installEventFilter(this);
if (column.columnInfo)
KexiImageContextMenu::updateTitle( d->popup, column.columnInfo->captionOrAliasOrName(),
//! @todo pixmaplabel icon is hardcoded...
"pixmaplabel" );
d->button->setPopup( d->popup );
//force edit requested to start editing... (this will call slotUpdateActionsAvailabilityRequested())
//connect(d->popup, TQT_SIGNAL(aboutToShow()), this, TQT_SIGNAL(editRequested()));
connect(d->popup, TQT_SIGNAL(updateActionsAvailabilityRequested(bool&, bool&)),
this, TQT_SLOT(slotUpdateActionsAvailabilityRequested(bool&, bool&)));
connect(d->popup, TQT_SIGNAL(insertFromFileRequested(const KURL&)),
this, TQT_SLOT(handleInsertFromFileAction(const KURL&)));
connect(d->popup, TQT_SIGNAL(saveAsRequested(const TQString&)),
this, TQT_SLOT(handleSaveAsAction(const TQString&)));
connect(d->popup, TQT_SIGNAL(cutRequested()),
this, TQT_SLOT(handleCutAction()));
connect(d->popup, TQT_SIGNAL(copyRequested()),
this, TQT_SLOT(handleCopyAction()));
connect(d->popup, TQT_SIGNAL(pasteRequested()),
this, TQT_SLOT(handlePasteAction()));
connect(d->popup, TQT_SIGNAL(clearRequested()),
this, TQT_SLOT(clear()));
connect(d->popup, TQT_SIGNAL(showPropertiesRequested()),
this, TQT_SLOT(handleShowPropertiesAction()));
}
KexiBlobTableEdit::~KexiBlobTableEdit()
{
delete d;
#if 0
kdDebug() << "KexiBlobTableEdit: Cleaning up..." << endl;
if (m_tempFile) {
m_tempFile->unlink();
//todo
}
delete m_proc;
m_proc = 0;
kdDebug() << "KexiBlobTableEdit: Ready." << endl;
#endif
}
//! initializes this editor with \a add value
void KexiBlobTableEdit::setValueInternal(const TQVariant& add, bool removeOld)
{
if (!d->setValueInternalEnabled)
return;
if (removeOld)
d->value = add.toByteArray();
else //do not add "m_origValue" to "add" as this is TQByteArray
d->value = m_origValue.toByteArray();
#if 0 //todo?
TQByteArray val = m_origValue.toByteArray();
kdDebug() << "KexiBlobTableEdit: Size of BLOB: " << val.size() << endl;
m_tempFile = new KTempFile();
m_tempFile->setAutoDelete(true);
kdDebug() << "KexiBlobTableEdit: Creating temporary file: " << m_tempFile->name() << endl;
m_tempFile->dataStream()->writeRawBytes(val.data(), val.size());
m_tempFile->close();
delete m_tempFile;
m_tempFile = 0;
KMimeMagicResult* mmr = KMimeMagic::self()->findFileType(m_tempFile->name());
kdDebug() << "KexiBlobTableEdit: Mimetype = " << mmr->mimeType() << endl;
setViewWidget( new TQWidget(this) );
#endif
}
bool KexiBlobTableEdit::valueIsNull()
{
//TODO
d->value.size();
return d->value.isEmpty();
}
bool KexiBlobTableEdit::valueIsEmpty()
{
//TODO
return d->value.isEmpty();
}
TQVariant
KexiBlobTableEdit::value()
{
return d->value;
#if 0
//todo
// ok = true;
if(m_content && m_content->isModified())
{
return TQVariant(m_content->text());
}
TQByteArray value;
TQFile f( m_tempFile->name() );
f.open(IO_ReadOnly);
TQDataStream stream(&f);
char* data = (char*) malloc(f.size());
value.resize(f.size());
stream.readRawBytes(data, f.size());
value.duplicate(data, f.size());
free(data);
kdDebug() << "KexiBlobTableEdit: Size of BLOB: " << value.size() << endl;
return TQVariant(value);
#endif
}
void KexiBlobTableEdit::paintFocusBorders( TQPainter *p, TQVariant &, int x, int y, int w, int h )
{
// d->currentEditorWidth = w;
if (!d->readOnly && w > d->button->width())
w -= d->button->width();
p->drawRect(x, y, w, h);
}
void
KexiBlobTableEdit::setupContents( TQPainter *p, bool focused, const TQVariant& val,
TQString &txt, int &align, int &x, int &y_offset, int &w, int &h )
{
Q_UNUSED(focused);
Q_UNUSED(txt);
Q_UNUSED(align);
//! @todo optimize: load to m_pixmap, downsize
TQPixmap pixmap;
x = 0;
w -= 1; //a place for border
h -= 1; //a place for border
if (p && val.canCast(TQVariant::ByteArray) && pixmap.loadFromData(val.toByteArray())) {
KexiUtils::drawPixmap( *p, 0/*lineWidth*/, TQRect(x, y_offset, w, h),
pixmap, TQt::AlignCenter, true/*scaledContents*/, true/*keepAspectRatio*/);
}
}
bool KexiBlobTableEdit::cursorAtStart()
{
return true;
}
bool KexiBlobTableEdit::cursorAtEnd()
{
return true;
}
void KexiBlobTableEdit::handleInsertFromFileAction(const KURL& url)
{
if (isReadOnly())
return;
TQString fileName( url.isLocalFile() ? url.path() : url.prettyURL() );
//! @todo download the file if remote, then set fileName properly
TQFile f(fileName);
if (!f.open(IO_ReadOnly)) {
//! @todo err msg
return;
}
TQByteArray ba = f.readAll();
if (f.status()!=IO_Ok) {
//! @todo err msg
f.close();
return;
}
f.close();
// m_valueMimeType = KImageIO::mimeType( fileName );
setValueInternal( ba, true );
signalEditRequested();
//emit acceptRequested();
}
void KexiBlobTableEdit::handleAboutToSaveAsAction(TQString& origFilename, TQString& fileExtension, bool& dataIsEmpty)
{
Q_UNUSED(origFilename);
Q_UNUSED(fileExtension);
dataIsEmpty = valueIsEmpty();
//! @todo no fname stored for now
}
void KexiBlobTableEdit::handleSaveAsAction(const TQString& fileName)
{
TQFile f(fileName);
if (!f.open(IO_WriteOnly)) {
//! @todo err msg
return;
}
f.writeBlock( d->value );
if (f.status()!=IO_Ok) {
//! @todo err msg
f.close();
return;
}
f.close();
}
void KexiBlobTableEdit::handleCutAction()
{
if (isReadOnly())
return;
handleCopyAction();
clear();
}
void KexiBlobTableEdit::handleCopyAction()
{
executeCopyAction(d->value);
}
void KexiBlobTableEdit::executeCopyAction(const TQByteArray& data)
{
TQPixmap pixmap;
if (!pixmap.loadFromData(data))
return;
tqApp->tqclipboard()->setPixmap(pixmap, TQClipboard::Clipboard);
}
void KexiBlobTableEdit::handlePasteAction()
{
if (isReadOnly())
return;
TQPixmap pm( tqApp->tqclipboard()->pixmap(TQClipboard::Clipboard) );
TQByteArray ba;
TQBuffer buffer( ba );
buffer.open( IO_WriteOnly );
if (pm.save( &buffer, "PNG" )) {// write pixmap into ba in PNG format
setValueInternal( ba, true );
}
else {
setValueInternal( TQByteArray(), true );
}
signalEditRequested();
//emit acceptRequested();
repaintRelatedCell();
}
void KexiBlobTableEdit::clear()
{
setValueInternal( TQByteArray(), true );
signalEditRequested();
//emit acceptRequested();
repaintRelatedCell();
}
void KexiBlobTableEdit::handleShowPropertiesAction()
{
//! @todo
}
void KexiBlobTableEdit::showFocus( const TQRect& r, bool readOnly )
{
d->readOnly = readOnly; //cache for slotUpdateActionsAvailabilityRequested()
// d->button->move( pos().x()+ width(), pos().y() );
updateFocus( r );
// d->button->setEnabled(!readOnly);
if (d->readOnly)
d->button->hide();
else
d->button->show();
}
void KexiBlobTableEdit::resize(int w, int h)
{
d->totalSize = TQSize(w,h);
const int addWidth = d->readOnly ? 0 : d->button->width();
TQWidget::resize(w - addWidth, h);
if (!d->readOnly)
d->button->resize( h, h );
m_rightMarginWhenFocused = m_rightMargin + addWidth;
TQRect r( pos().x(), pos().y(), w+1, h+1 );
r.moveBy(m_scrollView->contentsX(),m_scrollView->contentsY());
updateFocus( r );
//todo if (d->popup) {
//todo d->popup->updateSize();
//todo }
}
void KexiBlobTableEdit::updateFocus( const TQRect& r )
{
if (!d->readOnly) {
if (d->button->width() > r.width())
moveChild(d->button, r.right() + 1, r.top());
else
moveChild(d->button, r.right() - d->button->width(), r.top() );
}
}
void KexiBlobTableEdit::hideFocus()
{
d->button->hide();
}
TQSize KexiBlobTableEdit::totalSize() const
{
return d->totalSize;
}
void KexiBlobTableEdit::slotUpdateActionsAvailabilityRequested(bool& valueIsNull, bool& valueIsReadOnly)
{
emit editRequested();
valueIsNull = this->valueIsNull();
valueIsReadOnly = d->readOnly || isReadOnly();
}
void KexiBlobTableEdit::signalEditRequested()
{
d->setValueInternalEnabled = false;
emit editRequested();
d->setValueInternalEnabled = true;
}
bool KexiBlobTableEdit::handleKeyPress( TQKeyEvent* ke, bool editorActive )
{
Q_UNUSED(editorActive);
const int k = ke->key();
KKey kkey(ke);
if (!d->readOnly) {
if ((ke->state()==Qt::NoButton && k==TQt::Key_F4)
|| (ke->state()==TQt::AltButton && k==TQt::Key_Down)) {
d->button->animateClick();
TQMouseEvent me( TQEvent::MouseButtonPress, TQPoint(2,2), Qt::LeftButton, Qt::NoButton );
TQApplication::sendEvent( d->button, &me );
}
else if ((ke->state()==Qt::NoButton && (k==TQt::Key_F2 || k==TQt::Key_Space || k==TQt::Key_Enter || k==TQt::Key_Return))) {
d->popup->insertFromFile();
}
else
return false;
}
else
return false;
return true;
}
bool KexiBlobTableEdit::handleDoubleClick()
{
d->popup->insertFromFile();
return true;
}
void KexiBlobTableEdit::handleCopyAction(const TQVariant& value, const TQVariant& visibleValue)
{
Q_UNUSED(visibleValue);
executeCopyAction(value.toByteArray());
}
void KexiBlobTableEdit::handleAction(const TQString& actionName)
{
if (actionName=="edit_paste") {
d->popup->paste();
}
else if (actionName=="edit_cut") {
emit editRequested();
d->popup->cut();
}
}
bool KexiBlobTableEdit::eventFilter( TQObject *o, TQEvent *e )
{
if (TQT_BASE_OBJECT(o) == TQT_BASE_OBJECT(d->popup) && e->type()==TQEvent::KeyPress) {
TQKeyEvent* ke = TQT_TQKEYEVENT(e);
const int state = ke->state();
const int k = ke->key();
if ( (state==Qt::NoButton && (k==TQt::Key_Tab || k==TQt::Key_Left || k==TQt::Key_Right))
|| (state==TQt::ShiftButton && k==TQt::Key_Backtab)
)
{
d->popup->hide();
TQApplication::sendEvent( this, ke ); //re-send to move cursor
return true;
}
}
return false;
}
KEXI_CELLEDITOR_FACTORY_ITEM_IMPL(KexiBlobEditorFactoryItem, KexiBlobTableEdit)
//=======================
// KexiKIconTableEdit class is temporarily here:
//! @internal
class KexiKIconTableEdit::Private
{
public:
Private()
: pixmapCache(17, 17, false)
{
}
//! We've no editor widget that would store current value, so we do this here
TQVariant currentValue;
TQCache<TQPixmap> pixmapCache;
};
KexiKIconTableEdit::KexiKIconTableEdit(KexiTableViewColumn &column, TQWidget *parent)
: KexiTableEdit(column, parent)
, d( new Private() )
{
setName("KexiKIconTableEdit");
init();
}
KexiKIconTableEdit::~KexiKIconTableEdit()
{
delete d;
}
void KexiKIconTableEdit::init()
{
m_hasFocusableWidget = false;
d->pixmapCache.setAutoDelete(true);
}
void KexiKIconTableEdit::setValueInternal(const TQVariant& /*add*/, bool /*removeOld*/)
{
d->currentValue = m_origValue;
}
bool KexiKIconTableEdit::valueIsNull()
{
return d->currentValue.isNull();
}
bool KexiKIconTableEdit::valueIsEmpty()
{
return d->currentValue.isNull();
}
TQVariant KexiKIconTableEdit::value()
{
return d->currentValue;
}
void KexiKIconTableEdit::clear()
{
d->currentValue = TQVariant();
}
bool KexiKIconTableEdit::cursorAtStart()
{
return true;
}
bool KexiKIconTableEdit::cursorAtEnd()
{
return true;
}
void KexiKIconTableEdit::setupContents( TQPainter *p, bool /*focused*/, const TQVariant& val,
TQString &/*txt*/, int &/*align*/, int &/*x*/, int &y_offset, int &w, int &h )
{
Q_UNUSED( y_offset );
#if 0
#ifdef TQ_WS_WIN
y_offset = -1;
#else
y_offset = 0;
#endif
int s = TQMAX(h - 5, 12);
s = TQMIN( h-3, s );
s = TQMIN( w-3, s );//avoid too large box
TQRect r( TQMAX( w/2 - s/2, 0 ) , h/2 - s/2 /*- 1*/, s, s);
p->setPen(TQPen(tqcolorGroup().text(), 1));
p->drawRect(r);
if (val.asBool()) {
p->drawLine(r.x(), r.y(), r.right(), r.bottom());
p->drawLine(r.x(), r.bottom(), r.right(), r.y());
}
#endif
TQString key = val.toString();
TQPixmap *pix = 0;
if (!key.isEmpty() && !(pix = d->pixmapCache[ key ])) {
//cache pixmap
TQPixmap pm = KGlobal::iconLoader()->loadIcon( key, KIcon::Small,
0, KIcon::DefaultState, 0L, true/*canReturnNull*/ );
if (!pm.isNull()) {
pix = new TQPixmap(pm);
d->pixmapCache.insert(key, pix);
}
}
if (p && pix) {
p->drawPixmap( (w-pix->width())/2, (h-pix->height())/2, *pix );
}
}
void KexiKIconTableEdit::handleCopyAction(const TQVariant& value, const TQVariant& visibleValue)
{
Q_UNUSED(value);
Q_UNUSED(visibleValue);
}
KEXI_CELLEDITOR_FACTORY_ITEM_IMPL(KexiKIconTableEditorFactoryItem, KexiKIconTableEdit)
#include "kexiblobtableedit.moc"