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/kcoloredit/paletteviewscrolledarea.cpp

410 lines
13 KiB

/***************************************************************************
paletteviewscrolledarea.cpp
-------------------
begin : Sun Jul 17 2000
copyright : (C) 2000 by Artur Rataj
email : art@zeus.polsl.gliwice.pl
***************************************************************************/
/***************************************************************************
* *
* 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. *
* *
***************************************************************************/
#include <stdlib.h>
#include <tqptrlist.h>
#include <tqcolor.h>
#include <tqcursor.h>
#include <tqbrush.h>
#include <tqpainter.h>
#include <tqpixmap.h>
#include <tqpen.h>
#include <tqfontmetrics.h>
#include <tqtimer.h>
#include <kglobal.h>
#include <kcolordrag.h>
#include <tqscrollbar.h>
#include "palette.h"
#include "palettehistory.h"
#include "kcoloreditdoc.h"
#include "kcoloreditview.h"
#include "paletteviewscrolledarea.h"
#include "paletteviewscrolledarea.moc"
PaletteViewScrolledArea::PaletteViewScrolledArea(const int defaultCellWidth,
const int defaultCellHeight, const int cellSpacing, TQScrollBar* scrollBar,
TQScrollBar* hScrollBar, KColorEditView* view, TQWidget* parent, const char* name)
: TQFrame(parent, name) {
this->defaultCellWidth = defaultCellWidth;
this->defaultCellHeight = defaultCellHeight;
this->cellSpacing = cellSpacing;
this->scrollBar = scrollBar;
this->hScrollBar = hScrollBar;
this->view = view;
setBackgroundMode(NoBackground);
scrollTimeoutTimer = new TQTimer(this);
connect(scrollTimeoutTimer, TQT_SIGNAL( timeout() ), TQT_SLOT( slotScrollTimeout() ));
scrollTimeout = true;
mousePressed = false;
cursorPositioning = false;
colorChosen = false;
setCellsSizes();
slotCursorFollowsChosenColor(false);
this->scrollBar->setValue(0);
slotViewColorNames(false);
setMinimumSize(1, 1);
}
PaletteViewScrolledArea::~PaletteViewScrolledArea() {
}
void PaletteViewScrolledArea::slotCursorFollowsChosenColor(bool follows) {
cursorFollowsChosenColor = follows;
}
void PaletteViewScrolledArea::slotViewColorNames(bool viewColorNames) {
this->viewColorNames = viewColorNames;
}
void PaletteViewScrolledArea::redraw() {
//setCellsSizes();
repaintPalette();
}
void PaletteViewScrolledArea::repaintPalette() {
repaint(false);
}
void PaletteViewScrolledArea::checkSelectionAutoScroll(const int mousePosY) {
if(mousePosY < 0)
scrollPalette(-8, 50);
else if(mousePosY > height() - 1)
scrollPalette(8, 50);
}
void PaletteViewScrolledArea::scrollPalette(const int amount, const int timeout) {
if(scrollTimeout) {
scrollBy(amount);
scrollTimeout = false;
scrollTimeoutTimer->start(timeout, true);
}
}
void PaletteViewScrolledArea::slotScrollTimeout() {
scrollTimeout = true;
if(mousePressed) {
TQPoint cursorPoint = mapFromGlobal(TQCursor::pos());
setCursorPos(cursorPoint.x(), cursorPoint.y());
selectionEnd = cursorPos();
if(selectionEnd >= selectionBegin)
setSelection(selectionBegin, selectionEnd);
else
setSelection(selectionEnd, selectionBegin);
checkSelectionAutoScroll(cursorPoint.y());
repaintPalette();
}
}
void PaletteViewScrolledArea::setCursorPos(const int pos) {
int oldCursorPos = document()->paletteCursorPos();
document()->setPaletteCursorPos(pos);
if(pos != oldCursorPos)
view->slotCursorPosChanged(pos);
}
bool PaletteViewScrolledArea::setCursorPos(const int x, const int y) {
int shiftedY = y + scrollBar->value();
int shiftedX = x + hScrollBar->value();
int cursorColumn = (shiftedX + rowWidth/cellsInRow/2)*cellsInRow/rowWidth;
int cursorRow;
if(shiftedY >= 0)
cursorRow = (shiftedY + cellSpacing)/rowHeight;
else
cursorRow = -1;
bool atCursorLocation = abs(cursorColumn*rowWidth/cellsInRow - shiftedX) < cellSpacing + 1;
if(!atCursorLocation) {
atCursorLocation = abs(cursorRow*rowHeight - shiftedY) < cellSpacing + 1;
if(atCursorLocation)
cursorColumn = 0;
}
if(atCursorLocation || cursorPositioning) {
if(cursorColumn > cellsInRow - 1) {
cursorColumn = 0;
++cursorRow;
}
int tmpCursorPos = cursorRow*cellsInRow + cursorColumn;
if(tmpCursorPos < 0)
tmpCursorPos = 0;
else if(tmpCursorPos >= palette()->length()) {
if(tmpCursorPos == palette()->length() &&
cursorColumn == 0 &&
selectionMin() == selectionMax())
/* if cursor has been set after the last color and the color is
* also the last in a row, and there is no selection, scroll the
* palette to ensure the cursor is visible
*/
scrollBy(rowHeight*2);
if(tmpCursorPos > palette()->length())
tmpCursorPos = palette()->length();
}
setCursorPos(tmpCursorPos);
}
return atCursorLocation;
}
void PaletteViewScrolledArea::setSelection(const int min, const int max) {
document()->setPaletteSelection(min, max);
}
void PaletteViewScrolledArea::setCellsSizes() {
rowWidth = width();
cellWidth = defaultCellWidth;
cellHeight = defaultCellHeight;
if(viewColorNames) {
TQPainter painter;
painter.begin(this);
if(cellHeight < painter.fontMetrics().height()) {
int prevCellHeight = cellHeight;
cellHeight = painter.fontMetrics().height();
cellWidth = cellHeight*cellWidth/prevCellHeight;
}
painter.end();
}
if(viewColorNames)
cellsInRow = 1;
else
cellsInRow = rowWidth/(cellWidth + cellSpacing*2);
if(viewColorNames)
rowHeight = cellHeight + cellSpacing*2;
else
rowHeight = (int)(rowWidth*1.0/cellsInRow/
( cellWidth + cellSpacing*2 )*( cellHeight + cellSpacing*2 ) + 0.5);
cellHeight = rowHeight - cellSpacing*2;
rowsNum = (palette()->length() + cellsInRow - 1)/cellsInRow;
cellTableHeight = rowsNum*rowHeight;
int contentsHeight;
if(palette()->length() != 0 &&
( palette()->length()%cellsInRow ) == 0)
contentsHeight = cellTableHeight + rowHeight;
else
contentsHeight = cellTableHeight;
int scrollBarRange = contentsHeight - 1 - height();
if(scrollBarRange < 0)
scrollBarRange = 0;
scrollBar->setRange(0, scrollBarRange);
scrollBar->setSteps(rowHeight, height());
}
Palette* PaletteViewScrolledArea::palette() const {
return document()->paletteHistory()->editableStream();
}
int PaletteViewScrolledArea::cursorPos() const {
return document()->paletteCursorPos();
}
int PaletteViewScrolledArea::selectionMin() const{
return document()->paletteSelectionBegin();
}
int PaletteViewScrolledArea::selectionMax() const {
return document()->paletteSelectionEnd();
}
void PaletteViewScrolledArea::paintEvent(TQPaintEvent* /*event*/) {
setCellsSizes();
TQPixmap pixmap(size());
TQPainter painter;
painter.begin(TQT_TQPAINTDEVICE(&pixmap), this);
TQFontMetrics fontMetrics = painter.fontMetrics();
int maxLineWidth;
if(viewColorNames) {
int maxTextLength = 0;
for(int index = 0; index < palette()->length(); ++index) {
int currTextLength = fontMetrics.width(
palette()->color(index)->name());
if(currTextLength > maxTextLength)
maxTextLength = currTextLength;
}
maxLineWidth = cellWidth + cellSpacing*2 +
cellSpacing*3 + maxTextLength + 1;
} else
maxLineWidth = rowWidth;
int width = rowWidth;
if(maxLineWidth > width) {
hScrollBar->setRange(0, maxLineWidth - width);
hScrollBar->setSteps(8, width);
hScrollBar->show();
} else {
hScrollBar->setValue(0);
hScrollBar->hide();
}
int posY = scrollBar->value();
int firstRow = posY/rowHeight;
int lastRow = (posY + height() - 1 + rowHeight - 1)/rowHeight;
if(viewColorNames)
painter.fillRect(0, 0, rowWidth, height(),
TQBrush( TQFrame::palette().active().base() ));
TQBrush normalBackgroundBrush(TQFrame::palette().active().background());
TQBrush selectedBackgroundBrush(TQFrame::palette().active().highlight());
TQBrush foregroundBrush;
TQBrush cursorBrush(TQFrame::palette().active().foreground());
TQPen backgroundPen(TQFrame::palette().active().foreground());
int min = selectionMin();
int max = selectionMax();
int fontAscent = fontMetrics.ascent();
int xBegin = -hScrollBar->value();
for(int x = 0; x < cellsInRow; ++x) {
int xEnd = -hScrollBar->value();
if(viewColorNames)
xEnd += cellWidth + cellSpacing*2;
else
xEnd += (x + 1)*(width - 1)/cellsInRow;
int cellWithSpacingWidth = xEnd - xBegin + 1;
int cellWidth = cellWithSpacingWidth - 2*cellSpacing;
for(int y = firstRow; y <= lastRow; ++y) {
int yBegin = y*rowHeight - posY;
int currCellNum = y*cellsInRow + x;
TQBrush* backgroundBrush;
if(currCellNum >= min && currCellNum < max)
backgroundBrush = &selectedBackgroundBrush;
else
backgroundBrush = &normalBackgroundBrush;
if(currCellNum < palette()->length()) {
Color* color = palette()->color(currCellNum);
TQBrush foregroundBrush(TQColor(
color->component(Color::RED_INDEX),
color->component(Color::GREEN_INDEX),
color->component(Color::BLUE_INDEX) ));
painter.fillRect(xBegin, yBegin, cellWithSpacingWidth, cellSpacing,
*backgroundBrush);
painter.fillRect(xBegin, yBegin + rowHeight - cellSpacing, cellWithSpacingWidth, cellSpacing,
*backgroundBrush);
TQBrush* backgroundOrCursorBrush;
if(cursorPos() == currCellNum)
backgroundOrCursorBrush = &cursorBrush;
else
backgroundOrCursorBrush = backgroundBrush;
painter.fillRect(xBegin, yBegin + cellSpacing, cellSpacing, cellHeight,
*backgroundOrCursorBrush);
painter.fillRect(xBegin + cellWithSpacingWidth - cellSpacing, yBegin + cellSpacing,
cellSpacing, cellHeight,
*backgroundBrush);
painter.fillRect(xBegin + cellSpacing, yBegin + cellSpacing, cellWidth, cellHeight,
foregroundBrush);
if(viewColorNames) {
painter.setPen(backgroundPen);
painter.drawText(xBegin + cellWithSpacingWidth + cellSpacing*3,
yBegin + rowHeight/2 + fontAscent/2, color->name());
}
} else {
if(cursorPos() == currCellNum) {
painter.fillRect(xBegin, yBegin + cellSpacing, cellSpacing, cellHeight,
cursorBrush);
painter.fillRect(xBegin, yBegin, cellSpacing, cellSpacing,
*backgroundBrush);
painter.fillRect(xBegin, yBegin + rowHeight - cellSpacing, cellSpacing, cellSpacing,
*backgroundBrush);
painter.fillRect(xBegin + cellSpacing, yBegin, cellWithSpacingWidth - cellSpacing, rowHeight,
*backgroundBrush);
} else
painter.fillRect(xBegin, yBegin, cellWithSpacingWidth, rowHeight,
*backgroundBrush);
}
}
xBegin = xEnd + 1;
}
painter.end();
painter.begin(this);
painter.drawPixmap(0, 0, pixmap);
painter.end();
}
int PaletteViewScrolledArea::colorIndex(const TQPoint& point) const {
int colorColumn = point.x()*cellsInRow/rowWidth;
int colorRow = (point.y() + scrollBar->value())/rowHeight;
int colorIndex = colorRow*cellsInRow + colorColumn;
if(colorIndex > palette()->length() - 1 ||
colorIndex < 0)
colorIndex = -1;
return colorIndex;
}
TQColor PaletteViewScrolledArea::color(const TQPoint& point) const {
Color* color = palette()->color(colorIndex( point ));
return TQColor(color->component( Color::RED_INDEX ),
color->component( Color::GREEN_INDEX ),
color->component( Color::BLUE_INDEX ));
}
void PaletteViewScrolledArea::mousePressEvent(TQMouseEvent* event) {
cursorPositioning = false;
if(( cursorPositioning = setCursorPos(event->x(), event->y()) )) {
selectionBegin = cursorPos();
setSelection(selectionBegin, selectionBegin);
redraw();
colorChosen = false;
} else {
colorDragPoint = event->pos();
colorChosen = true;
}
mousePressed = true;
}
void PaletteViewScrolledArea::mouseMoveEvent(TQMouseEvent* event) {
if(cursorPositioning) {
setCursorPos(event->x(), event->y());
selectionEnd = cursorPos();
if(selectionEnd >= selectionBegin)
setSelection(selectionBegin, selectionEnd);
else
setSelection(selectionEnd, selectionBegin);
checkSelectionAutoScroll(event->y());
redraw();
} else {
/* check if it is a color drag */
if(colorIndex( colorDragPoint ) != -1) {
if(abs( event->x() - colorDragPoint.x() ) > 2 ||
abs( event->y() - colorDragPoint.y() ) > 2) {
TQColor draggedColor = color(colorDragPoint);
KColorDrag* colorDrag = KColorDrag::makeDrag(draggedColor, this);
colorDrag->dragCopy();
} else
colorChosen = true;
}
}
}
void PaletteViewScrolledArea::mouseReleaseEvent(TQMouseEvent* /*event*/) {
if(colorChosen) {
if(colorIndex( colorDragPoint ) != -1) {
int index = colorIndex(colorDragPoint);
chooseColor(palette()->color( index ));
if(cursorFollowsChosenColor) {
setCursorPos(index);
setSelection(cursorPos(), cursorPos());
redraw();
}
}
colorChosen = false;
}
mousePressed = false;
}
void PaletteViewScrolledArea::chooseColor(Color* const color) {
view->chooseColor(color);
}
KColorEditDoc* PaletteViewScrolledArea::document() const {
return view->document();
}
void PaletteViewScrolledArea::scrollBy(const int y) {
scrollBar->setValue(scrollBar->value() + y);
}