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.
tdegraphics/kiconedit/kicongrid.cpp

2264 lines
57 KiB

/*
TDE Icon Editor - a small graphics drawing program for the TDE.
Copyright (C) 1998 Thomas Tanghus (tanghus@kde.org)
Includes portions of code from TQt,
Copyright (C) 1992-2000 Trolltech AS.
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 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 <stdlib.h>
#include <tqpainter.h>
#include <tqwhatsthis.h>
#include <tqscrollview.h>
#include <tqbitmap.h>
#include <tqclipboard.h>
#include <tqdatetime.h>
#include <kiconloader.h>
#include <kruler.h>
#include <klocale.h>
#include <kmessagebox.h>
#include <kdebug.h>
#include "kresize.h"
#include "properties.h"
#include "kicongrid.h"
#include "kiconedit.h"
#ifndef PICS_INCLUDED
#include "pics/logo.xpm"
#define PICS_INCLUDED
#endif
#include <X11/Xos.h>
void DrawCommand::execute()
{
oldcolor = *((uint*)image->scanLine(y) + x);
*((uint*)image->scanLine(y) + x) = newcolor;
int cell = y * grid->numCols() + x;
grid->setUndoColor( cell, newcolor, false );
}
void DrawCommand::unexecute()
{
*((uint*)image->scanLine(y) + x) = oldcolor;
int cell = y * grid->numCols() + x;
grid->setUndoColor( cell, oldcolor, false );
}
void RepaintCommand::execute()
{
grid->update( area);
}
KGridView::KGridView(TQImage *image, KCommandHistory* history, TQWidget *parent, const char *name)
: TQFrame(parent, name)
{
_corner = 0L;
_hruler = _vruler = 0L;
_grid = 0L;
acceptdrop = false;
KIconEditProperties *props = KIconEditProperties::self();
viewport = new TQScrollView(this);
TQ_CHECK_PTR(viewport);
_grid = new KIconEditGrid(image, history, viewport->viewport());
TQ_CHECK_PTR(_grid);
viewport->addChild(_grid);
_grid->setGrid(props->showGrid());
_grid->setCellSize(props->gridScale());
TQString str = i18n( "Icon draw grid\n\nThe icon grid is the area where"
" you draw the icons.\nYou can zoom in and out using the magnifying"
" glasses on the toolbar.\n(Tip: Hold the magnify button down for a"
" few seconds to zoom to a predefined scale)" );
TQWhatsThis::add( _grid, str );
if(props->bgMode() == FixedPixmap)
{
TQPixmap pix(props->bgPixmap());
if(pix.isNull())
{
TQPixmap pmlogo((const char **)logo);
pix = pmlogo;
}
viewport->viewport()->setBackgroundPixmap(pix);
_grid->setBackgroundPixmap(pix);
}
else
{
viewport->viewport()->setBackgroundColor(props->bgColor());
}
_corner = new TQFrame(this);
_corner->setFrameStyle(TQFrame::WinPanel | TQFrame::Raised);
_hruler = new KRuler(Qt::Horizontal, this);
_hruler->setEndLabel(i18n("width"));
_hruler->setOffset( -2 );
_hruler->setRange(0, 1000);
_vruler = new KRuler(Qt::Vertical, this);
_vruler->setEndLabel(i18n("height"));
_vruler->setOffset( -2 );
_vruler->setRange(0, 1000);
str = i18n( "Rulers\n\nThis is a visual representation of the current"
" cursor position" );
TQWhatsThis::add( _hruler, str );
TQWhatsThis::add( _vruler, str );
connect(_grid, TQT_SIGNAL(scalingchanged(int)), TQT_SLOT(scalingChange(int)));
connect(_grid, TQT_SIGNAL(sizechanged(int, int)), TQT_SLOT(sizeChange(int, int)));
connect(_grid, TQT_SIGNAL(needPainting()), TQT_SLOT(paintGrid()));
connect( _grid, TQT_SIGNAL(xposchanged(int)), _hruler, TQT_SLOT(slotNewValue(int)) );
connect( _grid, TQT_SIGNAL(yposchanged(int)), _vruler, TQT_SLOT(slotNewValue(int)) );
connect(viewport, TQT_SIGNAL(contentsMoving(int, int)), TQT_SLOT(moving(int, int)));
setSizes();
TQResizeEvent e(size(), size());
resizeEvent(&e);
}
void KGridView::paintGrid()
{
_grid->update(viewRect());
}
void KGridView::setSizes()
{
if(KIconEditProperties::self()->showRulers())
{
_hruler->setLittleMarkDistance(_grid->scaling());
_vruler->setLittleMarkDistance(_grid->scaling());
_hruler->setMediumMarkDistance(5);
_vruler->setMediumMarkDistance(5);
_hruler->setBigMarkDistance(10);
_vruler->setBigMarkDistance(10);
_hruler->setShowTinyMarks(true);
_hruler->setShowLittleMarks(false);
_hruler->setShowMediumMarks(true);
_hruler->setShowBigMarks(true);
_hruler->setShowEndMarks(true);
_vruler->setShowTinyMarks(true);
_vruler->setShowLittleMarks(false);
_vruler->setShowMediumMarks(true);
_vruler->setShowBigMarks(true);
_vruler->setShowEndMarks(true);
_hruler->setPixelPerMark(_grid->scaling());
_vruler->setPixelPerMark(_grid->scaling());
_hruler->setMaxValue(_grid->width()+20);
_vruler->setMaxValue(_grid->height()+20);
_hruler->show();
_vruler->show();
_corner->show();
//resize(_grid->width()+_vruler->width(), _grid->height()+_hruler->height());
}
else
{
_hruler->hide();
_vruler->hide();
_corner->hide();
//resize(_grid->size());
}
}
void KGridView::sizeChange(int, int)
{
setSizes();
}
void KGridView::moving(int x, int y)
{
_hruler->setOffset(abs(x));
_vruler->setOffset(abs(y));
}
void KGridView::scalingChange(int)
{
setSizes();
}
void KGridView::setShowRulers(bool mode)
{
KIconEditProperties::self()->setShowRulers( mode );
setSizes();
TQResizeEvent e(size(), size());
resizeEvent(&e);
}
void KGridView::setAcceptDrop(bool a)
{
if(a == acceptdrop) return;
acceptdrop = a;
paintDropSite();
}
void KGridView::checkClipboard()
{
_grid->checkClipboard();
}
const TQRect KGridView::viewRect()
{
int x, y, cx, cy;
if(viewport->horizontalScrollBar()->isVisible())
{
x = viewport->contentsX();
cx = viewport->viewport()->width();
}
else
{
x = 0;
cx = viewport->contentsWidth();
}
if(viewport->verticalScrollBar()->isVisible())
{
y = viewport->contentsY();
cy = viewport->viewport()->height();
}
else
{
y = 0;
cy = viewport->contentsHeight();
}
return TQRect(x, y, cx, cy);
}
void KGridView::paintDropSite()
{
TQPainter p;
p.begin( _grid );
p.setRasterOp (NotROP);
p.drawRect(viewRect());
p.end();
}
void KGridView::paintEvent(TQPaintEvent *)
{
if(acceptdrop)
paintDropSite();
}
void KGridView::resizeEvent(TQResizeEvent*)
{
kdDebug(4640) << "KGridView::resizeEvent" << endl;
setSizes();
if(KIconEditProperties::self()->showRulers())
{
_hruler->setGeometry(_vruler->width(), 0, width(), _hruler->height());
_vruler->setGeometry(0, _hruler->height(), _vruler->width(), height());
_corner->setGeometry(0, 0, _vruler->width(), _hruler->height());
viewport->setGeometry(_corner->width(), _corner->height(),
width()-_corner->width(), height()-_corner->height());
}
else
viewport->setGeometry(0, 0, width(), height());
}
KIconEditGrid::KIconEditGrid(TQImage *image, KCommandHistory* h, TQWidget *parent, const char *name)
: KColorGrid(parent, name, 1)
{
img = image;
history = h;
selected = 0;
m_command = 0;
// the 42 normal kde colors - there can be an additional
// 18 custom colors in the custom colors palette
for(uint i = 0; i < 42; i++)
iconcolors.append(iconpalette[i]);
setupImageHandlers();
btndown = isselecting = ispasting = modified = false;
img->create(32, 32, 32);
img->setAlphaBuffer(true);
clearImage(img);
currentcolor = tqRgb(0,0,0)|OPAQUE_MASK;
emit colorSelected(currentcolor);
setMouseTracking(true);
setNumRows(32);
setNumCols(32);
fill(TRANSPARENT);
connect( kapp->clipboard(), TQT_SIGNAL(dataChanged()), TQT_SLOT(checkClipboard()));
connect( h, TQT_SIGNAL(commandExecuted()), this, TQT_SLOT(updatePreviewPixmap() ));
createCursors();
KIconEditProperties *props = KIconEditProperties::self();
setTransparencyDisplayType(props->transparencyDisplayType());
setTransparencySolidColor(props->transparencySolidColor());
setCheckerboardColor1(props->checkerboardColor1());
setCheckerboardColor2(props->checkerboardColor2());
setCheckerboardSize(props->checkerboardSize());
}
KIconEditGrid::~KIconEditGrid()
{
kdDebug(4640) << "KIconEditGrid - destructor: done" << endl;
}
void KIconEditGrid::paintEvent(TQPaintEvent *e)
{
const TQRect cellsRect(0, 0, numCols() * cellSize(), numRows() * cellSize());
const TQRect paintCellsRect = cellsRect.intersect(e->rect());
if(!paintCellsRect.isEmpty())
{
//TQTime time;
//time.start();
TQRgb *imageBuffer = new TQRgb[paintCellsRect.width() * paintCellsRect.height()];
const int cellsize = cellSize();
const int firstCellPixelsRemaining = cellsize - paintCellsRect.left() % cellsize;
if(transparencyDisplayType() == TRD_SOLIDCOLOR)
{
const TQRgb backgroundColor = transparencySolidColor().rgb();
const int backgroundRed = transparencySolidColor().red();
const int backgroundGreen = transparencySolidColor().green();
const int backgroundBlue = transparencySolidColor().blue();
const int firstCellX = paintCellsRect.left() / cellsize;
for(int y = paintCellsRect.top(); y <= paintCellsRect.bottom(); y++)
{
TQRgb *dest = imageBuffer + (y - paintCellsRect.top()) * paintCellsRect.width();
if(y % cellsize == 0 || dest == imageBuffer)
{
// Paint the first scanline in each block of cellSize() identical lines.
// The remaineder can just be copied from this one.
const int cellY = y / cellsize;
TQRgb *src = gridcolors.data() + cellY * numCols() + firstCellX;
TQRgb sourcePixel = *src++;
int sourceAlpha = tqAlpha(sourcePixel);
TQRgb c;
if(sourceAlpha == 255)
{
c = sourcePixel;
}
else
if(sourceAlpha == 0)
{
c = backgroundColor;
}
else
{
const int sourceRed = tqRed(sourcePixel);
const int sourceGreen = tqGreen(sourcePixel);
const int sourceBlue = tqBlue(sourcePixel);
int r = (sourceAlpha * (sourceRed - backgroundRed)) + 0x80;
r = backgroundRed + ((r + (r >> 8)) >> 8);
int g = (sourceAlpha * (sourceGreen - backgroundGreen)) + 0x80;
g = backgroundGreen + ((g + (g >> 8)) >> 8);
int b = (sourceAlpha * (sourceBlue - backgroundBlue)) + 0x80;
b = backgroundBlue + ((b + (b >> 8)) >> 8);
c = tqRgb(r, g, b);
}
int cellPixelsRemaining = firstCellPixelsRemaining;
for(int x = paintCellsRect.left(); x <= paintCellsRect.right(); x++)
{
if(cellPixelsRemaining == 0)
{
cellPixelsRemaining = cellsize;
// Fetch the next source pixel
sourcePixel = *src++;
sourceAlpha = tqAlpha(sourcePixel);
if(sourceAlpha == 255)
{
c = sourcePixel;
}
else
if(sourceAlpha == 0)
{
c = backgroundColor;
}
else
{
const int sourceRed = tqRed(sourcePixel);
const int sourceGreen = tqGreen(sourcePixel);
const int sourceBlue = tqBlue(sourcePixel);
//int r = backgroundRed + (sourceAlpha * (sourceRed - backgroundRed)) / 255;
//int g = backgroundGreen + (sourceAlpha * (sourceGreen - backgroundGreen)) / 255;
//int b = backgroundBlue + (sourceAlpha * (sourceBlue - backgroundBlue)) / 255;
int r = (sourceAlpha * (sourceRed - backgroundRed)) + 0x80;
r = backgroundRed + ((r + (r >> 8)) >> 8);
int g = (sourceAlpha * (sourceGreen - backgroundGreen)) + 0x80;
g = backgroundGreen + ((g + (g >> 8)) >> 8);
int b = (sourceAlpha * (sourceBlue - backgroundBlue)) + 0x80;
b = backgroundBlue + ((b + (b >> 8)) >> 8);
c = tqRgb(r, g, b);
}
}
cellPixelsRemaining--;
*dest++ = c;
}
}
else
{
// Copy the scanline above.
memcpy(dest, dest - paintCellsRect.width(), paintCellsRect.width() * sizeof(TQRgb));
}
}
}
else
{
int squareSize;
const int fixedPointMultiplier = 4;
if(checkerboardSize() == CHK_SMALL)
{
squareSize = (cellSize() * fixedPointMultiplier) / 4;
}
else
if(checkerboardSize() == CHK_MEDIUM)
{
squareSize = (cellSize() * fixedPointMultiplier) / 2;
}
else
{
squareSize = (2 * cellSize() * fixedPointMultiplier) / 2;
}
TQRgb *color1ScanLine = new TQRgb[paintCellsRect.width()];
TQRgb *color2ScanLine = new TQRgb[paintCellsRect.width()];
TQRgb *color1Buffer = color1ScanLine;
TQRgb *color2Buffer = color2ScanLine;
for(int x = paintCellsRect.left(); x <= paintCellsRect.right(); x++)
{
if((((x * fixedPointMultiplier) / squareSize) & 1) == 0)
{
*color1Buffer++ = checkerboardColor1().rgb();
*color2Buffer++ = checkerboardColor2().rgb();
}
else
{
*color1Buffer++ = checkerboardColor2().rgb();
*color2Buffer++ = checkerboardColor1().rgb();
}
}
const int firstCellX = paintCellsRect.left() / cellsize;
const int firstCellPixelsRemaining = cellsize - paintCellsRect.left() % cellsize;
int lastCellY = -1;
int lastLineFirstSquareColour = 0;
for(int y = paintCellsRect.top(); y <= paintCellsRect.bottom(); y++)
{
TQRgb *dest = imageBuffer + (y - paintCellsRect.top()) * paintCellsRect.width();
const int cellY = y / cellsize;
int firstSquareColour;
const TQRgb *checkerboardSrc;
if((((y * fixedPointMultiplier) / squareSize) & 1) == 0)
{
firstSquareColour = 1;
checkerboardSrc = color1ScanLine;
}
else
{
firstSquareColour = 2;
checkerboardSrc = color2ScanLine;
}
if(cellY == lastCellY && firstSquareColour == lastLineFirstSquareColour)
{
// Copy the scanline above.
memcpy(dest, dest - paintCellsRect.width(), paintCellsRect.width() * sizeof(TQRgb));
}
else
{
TQRgb *src = gridcolors.data() + cellY * numCols() + firstCellX;
TQRgb sourcePixel = *src++;
int sourceRed = tqRed(sourcePixel);
int sourceGreen = tqGreen(sourcePixel);
int sourceBlue = tqBlue(sourcePixel);
int sourceAlpha = tqAlpha(sourcePixel);
int cellPixelsRemaining = firstCellPixelsRemaining;
for(int x = paintCellsRect.left(); x <= paintCellsRect.right(); x++)
{
if(cellPixelsRemaining == 0)
{
cellPixelsRemaining = cellsize;
// Fetch the next source pixel
sourcePixel = *src++;
sourceRed = tqRed(sourcePixel);
sourceGreen = tqGreen(sourcePixel);
sourceBlue = tqBlue(sourcePixel);
sourceAlpha = tqAlpha(sourcePixel);
}
cellPixelsRemaining--;
TQRgb c;
if(sourceAlpha == 255)
{
c = sourcePixel;
}
else
if(sourceAlpha == 0)
{
c = *checkerboardSrc;
}
else
{
const int backgroundColor = *checkerboardSrc;
const int backgroundRed = tqRed(backgroundColor);
const int backgroundGreen = tqGreen(backgroundColor);
const int backgroundBlue = tqBlue(backgroundColor);
//int r = backgroundRed + (sourceAlpha * (sourceRed - backgroundRed)) / 255;
//int g = backgroundGreen + (sourceAlpha * (sourceGreen - backgroundGreen)) / 255;
//int b = backgroundBlue + (sourceAlpha * (sourceBlue - backgroundBlue)) / 255;
int r = (sourceAlpha * (sourceRed - backgroundRed)) + 0x80;
r = backgroundRed + ((r + (r >> 8)) >> 8);
int g = (sourceAlpha * (sourceGreen - backgroundGreen)) + 0x80;
g = backgroundGreen + ((g + (g >> 8)) >> 8);
int b = (sourceAlpha * (sourceBlue - backgroundBlue)) + 0x80;
b = backgroundBlue + ((b + (b >> 8)) >> 8);
c = tqRgb(r, g, b);
}
*dest++ = c;
checkerboardSrc++;
}
}
lastCellY = cellY;
lastLineFirstSquareColour = firstSquareColour;
}
delete [] color1ScanLine;
delete [] color2ScanLine;
}
TQImage image((uchar *)(imageBuffer), paintCellsRect.width(), paintCellsRect.height(), 32, 0, 0,
#if X_BYTE_ORDER == X_LITTLE_ENDIAN
TQImage::LittleEndian);
#else
TQImage::BigEndian);
#endif
Q_ASSERT(!image.isNull());
TQPixmap _pixmap;
_pixmap.convertFromImage(image);
TQPainter p;
p.begin(&_pixmap);
paintForeground(&p, e);
p.end();
bitBlt(this, paintCellsRect.left(), paintCellsRect.top(), &_pixmap);
//kdDebug(4640) << "Image render elapsed: " << time.elapsed() << endl;
delete [] imageBuffer;
}
}
void KIconEditGrid::paintForeground(TQPainter* p, TQPaintEvent* e)
{
TQWMatrix matrix;
matrix.translate(-e->rect().x(), -e->rect().y());
p->setWorldMatrix( matrix );
TQRect cellsRect(0, 0, numCols() * cellSize(), numRows() * cellSize());
TQRect paintCellsRect = cellsRect.intersect(e->rect());
if(!paintCellsRect.isEmpty())
{
int firstColumn = paintCellsRect.left() / cellSize();
int lastColumn = paintCellsRect.right() / cellSize();
int firstRow = paintCellsRect.top() / cellSize();
int lastRow = paintCellsRect.bottom() / cellSize();
p->setPen(TQColor(0, 0, 0));
p->setBrush(TQColor(0, 0, 0));
for(int column = firstColumn; column <= lastColumn; column++)
{
for(int row = firstRow; row <= lastRow; row++)
{
int x = column * cellSize();
int y = row * cellSize();
if((ispasting || isselecting) && isMarked(column, row))
{
p->drawWinFocusRect(x + 1, y + 1, cellSize() - 2, cellSize() - 2);
}
else
{
switch( tool )
{
case FilledRect:
case Rect:
case Ellipse:
case Circle:
case FilledEllipse:
case FilledCircle:
case Line:
if(btndown && isMarked(column, row))
{
if(cellSize() > 1)
{
p->drawWinFocusRect( x + 1, y + 1, cellSize() - 2, cellSize() - 2);
}
else
{
p->drawPoint(x, y);
}
}
break;
default:
break;
}
}
}
}
}
if(hasGrid()&& !(cellSize()==1))
{
p->setPen(TQColor(0, 0, 0));
int x = e->rect().x() - ((e->rect().x() % cellSize()) + cellSize());
if(x < 0) x = 0;
int y = e->rect().y() - ((e->rect().y() % cellSize()) + cellSize());
if(y < 0) y = 0;
int cx = e->rect().right() + cellSize();
int cy = e->rect().bottom() + cellSize();
// draw grid lines
for(int i = x; i < cx; i += cellSize())
p->drawLine(i, y, i, cy);
for(int i = y; i < cy; i += cellSize())
p->drawLine(x, i, cx, i);
}
}
void KIconEditGrid::mousePressEvent( TQMouseEvent *e )
{
if(!e || (e->button() != Qt::LeftButton))
return;
int row = findRow( e->pos().y() );
int col = findCol( e->pos().x() );
//int cell = row * numCols() + col;
if(!img->valid(col, row))
return;
btndown = true;
start.setX(col);
start.setY(row);
if(ispasting)
{
ispasting = false;
editPaste(true);
}
if(isselecting)
{
TQPointArray a(pntarray.copy());
pntarray.resize(0);
drawPointArray(a, Mark);
emit selecteddata(false);
}
switch( tool )
{
case SelectRect:
case SelectCircle:
isselecting = true;
break;
default:
break;
}
}
void KIconEditGrid::mouseMoveEvent( TQMouseEvent *e )
{
if(!e) return;
int row = findRow( e->pos().y() );
int col = findCol( e->pos().x() );
int cell = row * numCols() + col;
if(img->valid(col, row))
{
//kdDebug(4640) << col << " X " << row << endl;
emit poschanged(col, row);
// for the rulers
emit xposchanged((col*scaling())+scaling()/2);
emit yposchanged((row*scaling())+scaling()/2);
}
TQPoint tmpp(col, row);
if(tmpp == end) return;
// need to use intersection of rectangles to allow pasting
// only that part of clip image which intersects -jwc-
if(ispasting && !btndown && img->valid(col, row))
{
if( (col + cbsize.width()) > (numCols()-1) )
insrect.setX(numCols()-insrect.width());
else
insrect.setX(col);
if( (row + cbsize.height()) > (numRows()-1) )
insrect.setY(numRows()-insrect.height());
else
insrect.setY(row);
insrect.setSize(cbsize);
start = insrect.topLeft();
end = insrect.bottomRight();
drawRect(false);
return;
}
if(!img->valid(col, row) || !btndown)
return;
end.setX(col);
end.setY(row);
if(isselecting)
{
if(tool == SelectRect)
drawRect(false);
else
drawEllipse(false);
return;
}
bool erase=false;
switch( tool )
{
case Eraser:
erase=true;
case Freehand:
{
if( !m_command )
m_command = new KMacroCommand( i18n("Free Hand") );
if(erase)
setColor( cell, TRANSPARENT );
else
setColor( cell, currentcolor );
if ( selected != cell )
{
setModified( true );
int prevSel = selected;
selected = cell;
TQRect area = TQRect( col*cellsize,row*cellsize, cellsize, cellsize ).unite(
TQRect ( (prevSel%numCols())*cellsize,(prevSel/numCols())*cellsize, cellsize, cellsize ) );
m_command->addCommand( new RepaintCommand( area, this ) );
DrawCommand* dc = new DrawCommand( col, row, colorAt(cell), img, this );
RepaintCommand* rp = new RepaintCommand( area, this );
dc->execute();
rp->execute();
m_command->addCommand( dc );
m_command->addCommand( rp );
}
break;
}
case Find:
{
iconcolors.closestMatch(colorAt(cell));
if ( selected != cell )
{
int prevSel = selected;
selected = cell;
update((prevSel%numCols())*cellsize,(prevSel/numCols())*cellsize, cellsize, cellsize);
update(col*cellsize,row*cellsize, cellsize, cellsize);
emit colorSelected(colorAt(selected));
}
break;
}
case Ellipse:
case Circle:
case FilledEllipse:
case FilledCircle:
{
drawEllipse(false);
break;
}
case FilledRect:
case Rect:
{
drawRect(false);
break;
}
case Line:
{
drawLine(false, false);
break;
}
case Spray:
{
drawSpray(TQPoint(col, row));
setModified(true);
break;
}
default:
break;
}
p = *img;
emit changed(TQPixmap(p));
}
void KIconEditGrid::mouseReleaseEvent( TQMouseEvent *e )
{
if(!e || (e->button() != Qt::LeftButton))
return;
int row = findRow( e->pos().y() );
int col = findCol( e->pos().x() );
btndown = false;
end.setX(col);
end.setY(row);
int cell = row * numCols() + col;
bool erase=false;
switch( tool )
{
case Eraser:
erase=true;
//currentcolor = TRANSPARENT;
case Freehand:
{
if(!img->valid(col, row))
return;
if(erase)
setColor( cell, TRANSPARENT );
else
setColor( cell, currentcolor );
//if ( selected != cell )
//{
setModified( true );
int prevSel = selected;
selected = cell;
update((prevSel%numCols())*cellsize,(prevSel/numCols())*cellsize, cellsize, cellsize);
update(col*cellsize,row*cellsize, cellsize, cellsize);
//updateCell( prevSel/numCols(), prevSel%numCols(), FALSE );
//updateCell( row, col, FALSE );
*((uint*)img->scanLine(row) + col) = colorAt(cell);
p = *img;
//}
if( m_command ) {
history->addCommand( m_command, false );
m_command = 0;
}
break;
}
case Ellipse:
case Circle:
case FilledEllipse:
case FilledCircle:
{
drawEllipse(true);
break;
}
case FilledRect:
case Rect:
{
drawRect(true);
break;
}
case Line:
{
drawLine(true, false);
break;
}
case Spray:
{
drawSpray(TQPoint(col, row));
break;
}
case FloodFill:
{
TQApplication::setOverrideCursor(waitCursor);
drawFlood(col, row, colorAt(cell));
TQApplication::restoreOverrideCursor();
updateColors();
emit needPainting();
p = *img;
break;
}
case Find:
{
currentcolor = colorAt(cell);
if ( selected != cell )
{
int prevSel = selected;
selected = cell;
update((prevSel%numCols())*cellsize,(prevSel/numCols())*cellsize, cellsize, cellsize);
update(col*cellsize,row*cellsize, cellsize, cellsize);
emit colorSelected(currentcolor);
//updateCell( prevSel/numCols(), prevSel%numCols(), FALSE );
//updateCell( row, col, FALSE );
}
break;
}
default:
break;
}
emit changed(TQPixmap(p));
//emit colorschanged(numColors(), data());
}
//void KIconEditGrid::setColorSelection( const TQColor &color )
void KIconEditGrid::setColorSelection( uint c )
{
currentcolor = c;
emit colorSelected(currentcolor);
}
void KIconEditGrid::loadBlank( int w, int h )
{
img->create(w, h, 32);
img->setAlphaBuffer(true);
clearImage(img);
setNumRows(h);
setNumCols(w);
fill(TRANSPARENT);
emit sizechanged(numCols(), numRows());
emit colorschanged(numColors(), data());
history->clear();
}
void KIconEditGrid::load( TQImage *image)
{
kdDebug(4640) << "KIconEditGrid::load" << endl;
setUpdatesEnabled(false);
if(image == 0L)
{
TQString msg = i18n("There was an error loading a blank image.\n");
KMessageBox::error(this, msg);
return;
}
*img = image->convertDepth(32);
img->setAlphaBuffer(true);
setNumRows(img->height());
setNumCols(img->width());
for(int y = 0; y < numRows(); y++)
{
uint *l = (uint*)img->scanLine(y);
for(int x = 0; x < numCols(); x++, l++)
{
setColor((y*numCols())+x, *l, false);
}
//kdDebug(4640) << "Row: " << y << endl;
kapp->processEvents(200);
}
updateColors();
emit sizechanged(numCols(), numRows());
emit colorschanged(numColors(), data());
emit changed(pixmap());
setUpdatesEnabled(true);
emit needPainting();
//repaint(viewRect(), false);
history->clear();
}
const TQPixmap &KIconEditGrid::pixmap()
{
if(!img->isNull())
p = *img;
//p.convertFromImage(*img, 0);
return(p);
}
void KIconEditGrid::getImage(TQImage *image)
{
kdDebug(4640) << "KIconEditGrid::getImage" << endl;
*image = *img;
}
bool KIconEditGrid::zoomTo(int scale)
{
TQApplication::setOverrideCursor(waitCursor);
setUpdatesEnabled(false);
setCellSize( scale );
setUpdatesEnabled(true);
emit needPainting();
TQApplication::restoreOverrideCursor();
emit scalingchanged(cellSize());
if(scale == 1)
return false;
return true;
}
bool KIconEditGrid::zoom(Direction d)
{
int f = (d == DirIn) ? (cellSize()+1) : (cellSize()-1);
TQApplication::setOverrideCursor(waitCursor);
setUpdatesEnabled(false);
setCellSize( f );
setUpdatesEnabled(true);
//emit needPainting();
TQApplication::restoreOverrideCursor();
emit scalingchanged(cellSize());
if(d == DirOut && cellSize() <= 1)
return false;
return true;
}
void KIconEditGrid::checkClipboard()
{
bool ok = false;
TQImage tmp = clipboardImage(ok);
if(ok)
emit clipboarddata(true);
else
{
emit clipboarddata(false);
}
}
TQImage KIconEditGrid::clipboardImage(bool &ok)
{
//###### Remove me later.
//Workaround TQt bug -- check whether format provided first.
//Code below is from TQDragObject, to match the mimetype list....
TQStrList fileFormats = TQImageIO::inputFormats();
fileFormats.first();
bool oneIsSupported = false;
while ( fileFormats.current() )
{
TQCString format = fileFormats.current();
TQCString type = "image/" + format.lower();
if (kapp->clipboard()->data()->provides(type ) )
{
oneIsSupported = true;
}
fileFormats.next();
}
if (!oneIsSupported)
{
ok = false;
return TQImage();
}
TQImage image = kapp->clipboard()->image();
ok = !image.isNull();
if ( ok )
{
image = image.convertDepth(32);
image.setAlphaBuffer(true);
}
return image;
}
void KIconEditGrid::editSelectAll()
{
start.setX(0);
start.setY(0);
end.setX(numCols()-1);
end.setY(numRows()-1);
isselecting = true;
drawRect(false);
emit newmessage(i18n("All selected"));
}
void KIconEditGrid::editClear()
{
clearImage(img);
fill(TRANSPARENT);
update();
setModified(true);
p = *img;
emit changed(p);
emit newmessage(i18n("Cleared"));
}
TQImage KIconEditGrid::getSelection(bool cut)
{
const TQRect rect = pntarray.boundingRect();
int nx = 0, ny = 0, nw = 0, nh = 0;
rect.rect(&nx, &ny, &nw, &nh);
TQImage tmp(nw, nh, 32);
tmp.setAlphaBuffer(true);
clearImage(&tmp);
int s = pntarray.size();
for(int i = 0; i < s; i++)
{
int x = pntarray[i].x();
int y = pntarray[i].y();
if(img->valid(x, y) && rect.contains(TQPoint(x, y)))
{
*((uint*)tmp.scanLine(y-ny) + (x-nx)) = *((uint*)img->scanLine(y) + x);
if(cut)
{
*((uint*)img->scanLine(y) + x) = TRANSPARENT;
setColor( (y*numCols()) + x, TRANSPARENT, false );
}
}
}
TQPointArray a(pntarray.copy());
pntarray.resize(0);
drawPointArray(a, Mark);
emit selecteddata(false);
if(cut)
{
updateColors();
update(rect.x()*cellSize(), rect.y()*cellSize(),
rect.width()*cellSize(), rect.height()*cellSize());
p = *img;
emit changed(p);
emit colorschanged(numColors(), data());
emit newmessage(i18n("Selected area cut"));
setModified(true);
}
else
emit newmessage(i18n("Selected area copied"));
return tmp;
}
void KIconEditGrid::editCopy(bool cut)
{
kapp->clipboard()->setImage(getSelection(cut));
isselecting = false;
}
void KIconEditGrid::editPaste(bool paste)
{
bool ok = false;
TQImage tmp = clipboardImage(ok);
KIconEditProperties *props = KIconEditProperties::self();
if(ok)
{
if( (tmp.size().width() > img->size().width())
|| (tmp.size().height() > img->size().height()) )
{
if(KMessageBox::warningYesNo(this,
i18n("The clipboard image is larger than the current"
" image!\nPaste as new image?"),TQString(),i18n("Paste"), i18n("Do Not Paste")) == 0)
{
editPasteAsNew();
}
return;
}
else if(!paste)
{
ispasting = true;
cbsize = tmp.size();
return;
// emit newmessage(i18n("Pasting"));
}
else
{
//kdDebug(4640) << "KIconEditGrid: Pasting at: " << insrect.x() << " x " << insrect.y() << endl;
TQApplication::setOverrideCursor(waitCursor);
for(int y = insrect.y(), ny = 0; y < numRows() && ny < insrect.height(); y++, ny++)
{
uint *l = ((uint*)img->scanLine(y)+insrect.x());
uint *cl = (uint*)tmp.scanLine(ny);
for(int x = insrect.x(), nx = 0; x < numCols() && nx < insrect.width(); x++, nx++, l++, cl++)
{
if(props->pasteTransparent())
{
*l = *cl;
}
else
{
// Porter-Duff Over composition
double alphaS = tqAlpha(*cl) / 255.0;
double alphaD = tqAlpha(*l) / 255.0;
double r = tqRed(*cl) * alphaS + (1 - alphaS) * tqRed(*l) * alphaD;
double g = tqGreen(*cl) * alphaS + (1 - alphaS) * tqGreen(*l) * alphaD;
double b = tqBlue(*cl) * alphaS + (1 - alphaS) * tqBlue(*l) * alphaD;
double a = alphaS + (1 - alphaS) * alphaD;
// Remove multiplication by alpha
if(a > 0)
{
r /= a;
g /= a;
b /= a;
}
else
{
r = 0;
g = 0;
b = 0;
}
int ir = (int)(r + 0.5);
if(ir < 0)
{
ir = 0;
}
else
if(ir > 255)
{
ir = 255;
}
int ig = (int)(g + 0.5);
if(ig < 0)
{
ig = 0;
}
else
if(ig > 255)
{
ig = 255;
}
int ib = (int)(b + 0.5);
if(ib < 0)
{
ib = 0;
}
else
if(ib > 255)
{
ib = 255;
}
int ia = (int)((a * 255) + 0.5);
if(ia < 0)
{
ia = 0;
}
else
if(ia > 255)
{
ia = 255;
}
*l = tqRgba(ir, ig, ib, ia);
}
setColor((y*numCols())+x, (uint)*l, false);
}
}
updateColors();
update(insrect.x()*cellSize(), insrect.y()*cellSize(),
insrect.width()*cellSize(), insrect.height()*cellSize());
TQApplication::restoreOverrideCursor();
setModified(true);
p = *img;
emit changed(TQPixmap(p));
emit sizechanged(numCols(), numRows());
emit colorschanged(numColors(), data());
emit newmessage(i18n("Done pasting"));
}
}
else
{
TQString msg = i18n("Invalid pixmap data in clipboard!\n");
KMessageBox::sorry(this, msg);
}
}
void KIconEditGrid::editPasteAsNew()
{
bool ok = false;
TQImage tmp = clipboardImage(ok);
if(ok)
{
if(isModified())
{
KIconEdit *w = new KIconEdit(tmp);
TQ_CHECK_PTR(w);
}
else
{
*img = tmp;
load(img);
setModified(true);
//repaint(viewRect(), false);
p = *img;
emit changed(TQPixmap(p));
emit sizechanged(numCols(), numRows());
emit colorschanged(numColors(), data());
emit newmessage(i18n("Done pasting"));
history->clear();
}
}
else
{
TQString msg = i18n("Invalid pixmap data in clipboard!\n");
KMessageBox::error(this, msg);
}
}
void KIconEditGrid::editResize()
{
kdDebug(4640) << "KIconGrid::editResize" << endl;
KResizeDialog *rs = new KResizeDialog(this, 0, TQSize(numCols(), numRows()));
if(rs->exec())
{
const TQSize s = rs->getSize();
*img = img->smoothScale(s.width(), s.height());
load(img);
setModified(true);
}
delete rs;
}
void KIconEditGrid::setSize(const TQSize s)
{
kdDebug(4640) << "::setSize: " << s.width() << " x " << s.height() << endl;
img->create(s.width(), s.height(), 32);
img->setAlphaBuffer(true);
clearImage(img);
load(img);
}
void KIconEditGrid::createCursors()
{
TQBitmap mask(22, 22);
TQPixmap pix;
cursor_normal = TQCursor(arrowCursor);
pix = BarIcon("colorpicker-cursor");
if(pix.isNull())
{
cursor_colorpicker = cursor_normal;
kdDebug(4640) << "KIconEditGrid: Error loading colorpicker-cursor.xpm" << endl;
}
else
{
mask = TQPixmap(pix.createHeuristicMask());
pix.setMask(mask);
cursor_colorpicker = TQCursor(pix, 1, 21);
}
pix = BarIcon("paintbrush-cursor");
if(pix.isNull())
{
cursor_paint = cursor_normal;
kdDebug(4640) << "KIconEditGrid: Error loading paintbrush.xpm" << endl;
}
else
{
mask = TQPixmap(pix.createHeuristicMask());
pix.setMask(mask);
cursor_paint = TQCursor(pix, 0, 19);
}
pix = BarIcon("fill-cursor");
if(pix.isNull())
{
cursor_flood = cursor_normal;
kdDebug(4640) << "KIconEditGrid: Error loading fill-cursor.xpm" << endl;
}
else
{
mask = TQPixmap(pix.createHeuristicMask());
pix.setMask(mask);
cursor_flood = TQCursor(pix, 3, 20);
}
pix = BarIcon("aim-cursor");
if(pix.isNull())
{
cursor_aim = cursor_normal;
kdDebug(4640) << "KIconEditGrid: Error loading aim-cursor.xpm" << endl;
}
else
{
mask = TQPixmap(pix.createHeuristicMask());
pix.setMask(mask);
cursor_aim = TQCursor(pix, 10, 10);
}
pix = BarIcon("airbrush-cursor");
if(pix.isNull())
{
cursor_spray = cursor_normal;
kdDebug(4640) << "KIconEditGrid: Error loading airbrush-cursor.xpm" << endl;
}
else
{
mask = TQPixmap(pix.createHeuristicMask(true));
pix.setMask(mask);
cursor_spray = TQCursor(pix, 0, 20);
}
pix = BarIcon("eraser-cursor");
if(pix.isNull())
{
cursor_erase = cursor_normal;
kdDebug(4640) << "KIconEditGrid: Error loading eraser-cursor.xpm" << endl;
}
else
{
mask = TQPixmap(pix.createHeuristicMask(true));
pix.setMask(mask);
cursor_erase = TQCursor(pix, 1, 16);
}
}
void KIconEditGrid::setTool(DrawTool t)
{
btndown = false;
tool = t;
if(tool != SelectRect && tool != SelectCircle)
isselecting = false;
switch( tool )
{
case SelectRect:
isselecting = true;
setCursor(cursor_aim);
break;
case SelectCircle:
isselecting = true;
setCursor(cursor_aim);
break;
case Line:
case Ellipse:
case Circle:
case FilledEllipse:
case FilledCircle:
case FilledRect:
case Rect:
setCursor(cursor_aim);
break;
case Freehand:
setCursor(cursor_paint);
break;
case Spray:
setCursor(cursor_spray);
break;
case Eraser:
setCursor(cursor_erase);
break;
case FloodFill:
setCursor(cursor_flood);
break;
case Find:
setCursor(cursor_colorpicker);
break;
default:
break;
}
}
void KIconEditGrid::drawFlood(int x, int y, uint oldcolor)
{
if((!img->valid(x, y))
|| (colorAt((y * numCols())+x) != oldcolor)
|| (colorAt((y * numCols())+x) == currentcolor))
return;
*((uint*)img->scanLine(y) + x) = currentcolor;
setColor((y*numCols())+x, currentcolor, false);
setModified(true);
drawFlood(x, y-1, oldcolor);
drawFlood(x, y+1, oldcolor);
drawFlood(x-1, y, oldcolor);
drawFlood(x+1, y, oldcolor);
//TODO: add undo
}
void KIconEditGrid::drawSpray(TQPoint point)
{
int x = (point.x()-5);
int y = (point.y()-5);
//kdDebug(4640) << "drawSpray() - " << x << " X " << y << endl;
pntarray.resize(0);
int points = 0;
for(int i = 1; i < 4; i++, points++)
{
int dx = (rand() % 10);
int dy = (rand() % 10);
pntarray.putPoints(points, 1, x+dx, y+dy);
}
drawPointArray(pntarray, Draw);
}
//This routine is from TQt sources -- it's the branch of TQPointArray::makeEllipse( int x, int y, int w, int h ) that's not normally compiled
//It seems like KIconEdit relied on the TQt1 semantics for makeEllipse, which broke
//the tool with reasonably recent TQt versions.
//Thankfully, TQt includes the old code #ifdef'd, which is hence included here
static void TQPA_makeEllipse(TQPointArray& ar, int x, int y, int w, int h )
{ // midpoint, 1/4 ellipse
if ( w <= 0 || h <= 0 ) {
if ( w == 0 || h == 0 ) {
ar.resize( 0 );
return;
}
if ( w < 0 ) { // negative width
w = -w;
x -= w;
}
if ( h < 0 ) { // negative height
h = -h;
y -= h;
}
}
int s = (w+h+2)/2; // max size of xx,yy array
int *px = new int[s]; // 1/4th of ellipse
int *py = new int[s];
int xx, yy, i=0;
double d1, d2;
double a2=(w/2)*(w/2), b2=(h/2)*(h/2);
xx = 0;
yy = int(h/2);
d1 = b2 - a2*(h/2) + 0.25*a2;
px[i] = xx;
py[i] = yy;
i++;
while ( a2*(yy-0.5) > b2*(xx+0.5) ) { // region 1
if ( d1 < 0 ) {
d1 = d1 + b2*(3.0+2*xx);
xx++;
} else {
d1 = d1 + b2*(3.0+2*xx) + 2.0*a2*(1-yy);
xx++;
yy--;
}
px[i] = xx;
py[i] = yy;
i++;
}
d2 = b2*(xx+0.5)*(xx+0.5) + a2*(yy-1)*(yy-1) - a2*b2;
while ( yy > 0 ) { // region 2
if ( d2 < 0 ) {
d2 = d2 + 2.0*b2*(xx+1) + a2*(3-2*yy);
xx++;
yy--;
} else {
d2 = d2 + a2*(3-2*yy);
yy--;
}
px[i] = xx;
py[i] = yy;
i++;
}
s = i;
ar.resize( 4*s ); // make full point array
x += w/2;
y += h/2;
for ( i=0; i<s; i++ ) { // mirror
xx = px[i];
yy = py[i];
ar.setPoint( s-i-1, x+xx, y-yy );
ar.setPoint( s+i, x-xx, y-yy );
ar.setPoint( 3*s-i-1, x-xx, y+yy );
ar.setPoint( 3*s+i, x+xx, y+yy );
}
delete[] px;
delete[] py;
}
void KIconEditGrid::drawEllipse(bool drawit)
{
if(drawit)
{
drawPointArray(pntarray, Draw);
p = *img;
emit changed(p);
return;
}
TQPointArray a(pntarray.copy());
int x = start.x(), y = start.y(), cx, cy;
if(x > end.x())
{
cx = x - end.x();
x = x - cx;
}
else
cx = end.x() - x;
if(y > end.y())
{
cy = y - end.y();
y = y - cy;
}
else
cy = end.y() - y;
int d = (cx > cy) ? cx : cy;
//kdDebug(4640) << x << ", " << y << " - " << d << " " << d << endl;
pntarray.resize(0);
drawPointArray(a, Mark);
if(tool == Circle || tool == FilledCircle || tool == SelectCircle)
TQPA_makeEllipse(pntarray, x, y, d, d);
else if(tool == Ellipse || tool == FilledEllipse)
TQPA_makeEllipse(pntarray, x, y, cx, cy);
if((tool == FilledEllipse) || (tool == FilledCircle)
|| (tool == SelectCircle))
{
int s = pntarray.size();
int points = s;
for(int i = 0; i < s; i++)
{
int x = pntarray[i].x();
int y = pntarray[i].y();
for(int j = 0; j < s; j++)
{
if((pntarray[j].y() == y) && (pntarray[j].x() > x))
{
for(int k = x; k < pntarray[j].x(); k++, points++)
pntarray.putPoints(points, 1, k, y);
break;
}
}
}
}
drawPointArray(pntarray, Mark);
if(tool == SelectCircle && pntarray.size() > 0 && !ispasting)
emit selecteddata(true);
}
void KIconEditGrid::drawRect(bool drawit)
{
if(drawit)
{
drawPointArray(pntarray, Draw);
p = *img;
emit changed(p);
return;
}
TQPointArray a(pntarray.copy());
int x = start.x(), y = start.y(), cx, cy;
if(x > end.x())
{
cx = x - end.x();
x = x - cx;
}
else
cx = end.x() - x;
if(y > end.y())
{
cy = y - end.y();
y = y - cy;
}
else
cy = end.y() - y;
//kdDebug(4640) << x << ", " << y << " - " << cx << " " << cy << endl;
pntarray.resize(0);
drawPointArray(a, Mark); // remove previous marking
int points = 0;
bool pasting = ispasting;
if(tool == FilledRect || (tool == SelectRect))
{
for(int i = x; i <= x + (pasting ? cx + 1 : cx); i++)
{
for(int j = y; j <= y+cy; j++, points++)
pntarray.putPoints(points, 1, i, j);
}
}
else
{
for(int i = x; i <= x+cx; i++, points++)
pntarray.putPoints(points, 1, i, y);
for(int i = y; i <= y+cy; i++, points++)
pntarray.putPoints(points, 1, x, i);
for(int i = x; i <= x+cx; i++, points++)
pntarray.putPoints(points, 1, i, y+cy);
for(int i = y; i <= y+cy; i++, points++)
pntarray.putPoints(points, 1, x+cx, i);
}
drawPointArray(pntarray, Mark);
if(tool == SelectRect && pntarray.size() > 0 && !ispasting)
emit selecteddata(true);
}
void KIconEditGrid::drawLine(bool drawit, bool drawStraight)
{
if(drawit)
{
drawPointArray(pntarray, Draw);
p = *img;
emit changed(p);
return;
}
TQPointArray a(pntarray.copy());
pntarray.resize(0);
// remove previous marking
drawPointArray(a, Mark);
int x, y, dx, dy, delta;
dx = end.x() - start.x();
dy = end.y() - start.y();
x = start.x();
y = start.y();
delta = TQMAX(abs(dx), abs(dy));
int deltaX = abs(dx);
int deltaY = abs(dy);
if ((drawStraight) && (delta > 0))
{
dx /= delta;
dy /= delta;
for(int i = 0; i <= delta; i++)
{
pntarray.putPoints(i, 1, x, y);
x += dx;
y += dy;
}
}
else if ((delta > 0) && (deltaX >= deltaY))
{
for(int i = 0; i <= deltaX; i++)
{
pntarray.putPoints(i, 1, x, y);
if(dx > 0)
x++;
else
x--;
if(dy >= 0)
y = start.y() + (abs(start.x() - x) * deltaY) / deltaX;
else
y = start.y() - (abs(start.x() - x) * deltaY) / deltaX;
}
}
else if ((delta > 0) && (deltaY > deltaX))
{
for(int i = 0; i <= deltaY; i++)
{
pntarray.putPoints(i, 1, x, y);
if(dy > 0)
y++;
else
y--;
if(dx >= 0)
x = start.x() + (abs(start.y() - y) * deltaX) / deltaY;
else
x = start.x() - (abs(start.y() - y) * deltaX) / deltaY;
}
}
drawPointArray(pntarray, Mark);
}
void KIconEditGrid::drawPointArray(TQPointArray a, DrawAction action)
{
TQRect area( a.boundingRect().x()*cellSize()-1, a.boundingRect().y()*cellSize()-1,
a.boundingRect().width()*cellSize()+1, a.boundingRect().height()*cellSize()+1 );
KMacroCommand* macro = 0;
bool doupdate = false;
if( a.size() > 0 && action == Draw ) {
// might cause a memmory leak, if
// macro is never used and never
// added to the history! TODO: Fix this
macro = new KMacroCommand( i18n("Drawn Array") );
RepaintCommand* rc = new RepaintCommand( area, this );
macro->addCommand( rc );
}
int s = a.size(); //((rect.size().width()) * (rect.size().height()));
for(int i = 0; i < s; i++)
{
int x = a[i].x();
int y = a[i].y();
if(img->valid(x, y) && TQT_TQRECT_OBJECT(a.boundingRect()).contains(a[ i ]))
{
//kdDebug(4640) << "x: " << x << " - y: " << y << endl;
switch( action )
{
case Draw:
{
DrawCommand* dc = new DrawCommand( x, y, currentcolor, img, this );
dc->execute();
//*((uint*)img->scanLine(y) + x) = currentcolor; //colors[cell]|OPAQUE;
//int cell = y * numCols() + x;
//setColor( cell, currentcolor, false );
doupdate = true;
//updateCell( y, x, FALSE );
macro->addCommand( dc );
break;
}
case Mark:
case UnMark:
update(x*cellsize,y*cellsize, cellsize, cellsize);
//updateCell( y, x, true );
break;
default:
break;
}
}
}
if(doupdate)
{
setModified( true );
updateColors();
RepaintCommand* rc = new RepaintCommand( area, this );
rc->execute();
macro->addCommand( rc );
pntarray.resize(0);
// add to undo/redo history
history->addCommand( macro, false ); }
}
void KIconEditGrid::updatePreviewPixmap()
{
p = *img;
emit changed(TQPixmap(p));
}
bool KIconEditGrid::isMarked(TQPoint point)
{
return isMarked(point.x(), point.y());
}
bool KIconEditGrid::isMarked(int x, int y)
{
if(((y * numCols()) + x) == selected)
return true;
int s = pntarray.size();
for(int i = 0; i < s; i++)
{
if(y == pntarray[i].y() && x == pntarray[i].x())
return true;
}
return false;
}
// Fast diffuse dither to 3x3x3 color cube
// Based on TQt's image conversion functions
static bool kdither_32_to_8( const TQImage *src, TQImage *dst )
{
register TQRgb *p;
uchar *b;
int y;
//printf("kconvert_32_to_8\n");
if ( !dst->create(src->width(), src->height(), 8, 256) ) {
kdWarning() << "OImage: destination image not valid" << endl;
return FALSE;
}
int ncols = 256;
static uint bm[16][16];
static int init=0;
if (!init)
{
// Build a Bayer Matrix for dithering
init = 1;
int n, i, j;
bm[0][0]=0;
for (n=1; n<16; n*=2)
{
for (i=0; i<n; i++)
{
for (j=0; j<n; j++)
{
bm[i][j]*=4;
bm[i+n][j]=bm[i][j]+2;
bm[i][j+n]=bm[i][j]+3;
bm[i+n][j+n]=bm[i][j]+1;
}
}
}
for (i=0; i<16; i++)
for (j=0; j<16; j++)
bm[i][j]<<=8;
}
dst->setNumColors( ncols );
#define MAX_R 2
#define MAX_G 2
#define MAX_B 2
#define INDEXOF(r,g,b) (((r)*(MAX_G+1)+(g))*(MAX_B+1)+(b))
int rc, gc, bc;
for ( rc=0; rc<=MAX_R; rc++ ) // build 2x2x2 color cube
for ( gc=0; gc<=MAX_G; gc++ )
for ( bc=0; bc<=MAX_B; bc++ )
{
dst->setColor( INDEXOF(rc,gc,bc),
tqRgb( rc*255/MAX_R, gc*255/MAX_G, bc*255/MAX_B ) );
}
int sw = src->width();
int* line1[3];
int* line2[3];
int* pv[3];
line1[0] = new int[src->width()];
line2[0] = new int[src->width()];
line1[1] = new int[src->width()];
line2[1] = new int[src->width()];
line1[2] = new int[src->width()];
line2[2] = new int[src->width()];
pv[0] = new int[sw];
pv[1] = new int[sw];
pv[2] = new int[sw];
for ( y=0; y < src->height(); y++ )
{
p = (TQRgb *)src->scanLine(y);
b = dst->scanLine(y);
int endian = (TQImage::systemByteOrder() == TQImage::BigEndian);
int x;
uchar* q = const_cast<TQImage*>(src)->scanLine(y);
uchar* q2 = const_cast<TQImage*>(src)->scanLine(y+1 < src->height() ? y + 1 : 0);
for (int chan = 0; chan < 3; chan++)
{
b = dst->scanLine(y);
int *l1 = (y&1) ? line2[chan] : line1[chan];
int *l2 = (y&1) ? line1[chan] : line2[chan];
if ( y == 0 )
{
for (int i=0; i<sw; i++)
l1[i] = q[i*4+chan+endian];
}
if ( y+1 < src->height() )
{
for (int i=0; i<sw; i++)
l2[i] = q2[i*4+chan+endian];
}
// Bi-directional error diffusion
if ( y&1 )
{
for (x=0; x<sw; x++)
{
int pix = TQMAX(TQMIN(2, (l1[x] * 2 + 128)/ 255), 0);
int err = l1[x] - pix * 255 / 2;
pv[chan][x] = pix;
// Spread the error around...
if ( x+1<sw )
{
l1[x+1] += (err*7)>>4;
l2[x+1] += err>>4;
}
l2[x]+=(err*5)>>4;
if (x>1)
l2[x-1]+=(err*3)>>4;
}
}
else
{
for (x=sw; x-->0; )
{
int pix = TQMAX(TQMIN(2, (l1[x] * 2 + 128)/ 255), 0);
int err = l1[x] - pix * 255 / 2;
pv[chan][x] = pix;
// Spread the error around...
if ( x > 0 )
{
l1[x-1] += (err*7)>>4;
l2[x-1] += err>>4;
}
l2[x]+=(err*5)>>4;
if (x+1 < sw)
l2[x+1]+=(err*3)>>4;
}
}
}
if (endian)
{
for (x=0; x<sw; x++)
{
*b++ = INDEXOF(pv[2][x],pv[1][x],pv[0][x]);
}
}
else
{
for (x=0; x<sw; x++)
{
*b++ = INDEXOF(pv[0][x],pv[1][x],pv[2][x]);
}
}
}
delete [] line1[0];
delete [] line2[0];
delete [] line1[1];
delete [] line2[1];
delete [] line1[2];
delete [] line2[2];
delete [] pv[0];
delete [] pv[1];
delete [] pv[2];
#undef MAX_R
#undef MAX_G
#undef MAX_B
#undef INDEXOF
return TRUE;
}
// this doesn't work the way it should but the way KPixmap does.
void KIconEditGrid::mapToKDEPalette()
{
TQImage dest;
kdither_32_to_8(img, &dest);
*img = dest.convertDepth(32);
for(int y = 0; y < img->height(); y++)
{
uint *l = (uint*)img->scanLine(y);
for(int x = 0; x < img->width(); x++, l++)
{
if(*l < 0xff000000)
{
*l = *l | 0xff000000;
}
}
}
load(img);
return;
/*
#if [[[TQT_VERSION IS DEPRECATED]]] > 140
*img = img->convertDepthWithPalette(32, iconpalette, 42);
load(img);
return;
#endif
*/
TQApplication::setOverrideCursor(waitCursor);
for(int y = 0; y < numRows(); y++)
{
uint *l = (uint*)img->scanLine(y);
for(int x = 0; x < numCols(); x++, l++)
{
if(*l != TRANSPARENT)
{
if(!iconcolors.contains(*l))
*l = iconcolors.closestMatch(*l);
}
}
}
load(img);
setModified(true);
TQApplication::restoreOverrideCursor();
}
void KIconEditGrid::grayScale()
{
for(int y = 0; y < numRows(); y++)
{
uint *l = (uint*)img->scanLine(y);
for(int x = 0; x < numCols(); x++, l++)
{
if(*l != TRANSPARENT)
{
uint c = tqGray(*l);
*l = tqRgba(c, c, c, tqAlpha(*l));
}
}
}
load(img);
setModified(true);
}
void KIconEditGrid::clearImage(TQImage *image)
{
if(image->depth() != 32)
{
image->fill(TRANSPARENT);
}
else
{
// TQImage::fill() does not set the alpha channel so do it
// manually.
for(int y = 0; y < image->height(); y++)
{
uint *l = (uint*)image->scanLine(y);
for(int x = 0; x < image->width(); x++, l++)
{
*l = TRANSPARENT;
}
}
}
}
void KIconEditGrid::setModified(bool m)
{
if(m != modified)
{
modified = m;
emit modifiedchanged(m);
}
}
#include "kicongrid.moc"
// vim: set ts=4: