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.
256 lines
8.8 KiB
256 lines
8.8 KiB
/*
|
|
* kis_tool_duplicate.cc - part of Chalk
|
|
*
|
|
* Copyright (c) 2004 Cyrille Berger <cberger@cberger.net>
|
|
*
|
|
* 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 General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include <tqbitmap.h>
|
|
#include <tqcheckbox.h>
|
|
#include <tqpainter.h>
|
|
#include <tqlabel.h>
|
|
|
|
#include <kaction.h>
|
|
#include <kdebug.h>
|
|
#include <knuminput.h>
|
|
#include <klocale.h>
|
|
|
|
#include "kis_brush.h"
|
|
#include "kis_cursor.h"
|
|
#include "kis_image.h"
|
|
#include "kis_tool_duplicate.h"
|
|
#include "kis_painter.h"
|
|
#include "kis_vec.h"
|
|
#include "kis_button_press_event.h"
|
|
#include "kis_button_release_event.h"
|
|
#include "kis_move_event.h"
|
|
#include "kis_paintop.h"
|
|
#include "kis_paintop_registry.h"
|
|
#include "kis_canvas_subject.h"
|
|
#include "kis_perspective_grid.h"
|
|
|
|
#include "kis_canvas_painter.h"
|
|
#include "kis_boundary_painter.h"
|
|
|
|
KisToolDuplicate::KisToolDuplicate()
|
|
: super(i18n("Duplicate Brush")), m_isOffsetNotUptodate(true), m_position(TQPoint(-1,-1))
|
|
{
|
|
setName("tool_duplicate");
|
|
m_subject = 0;
|
|
setCursor(KisCursor::load("tool_duplicate_cursor.png", 5, 5));
|
|
}
|
|
|
|
KisToolDuplicate::~KisToolDuplicate()
|
|
{
|
|
}
|
|
|
|
void KisToolDuplicate::activate()
|
|
{
|
|
m_position = TQPoint(-1,-1);
|
|
super::activate();
|
|
if( m_subject->currentImg()->perspectiveGrid()->countSubGrids() != 1 )
|
|
{
|
|
m_perspectiveCorrection->setEnabled( false );
|
|
m_perspectiveCorrection->setChecked( false );
|
|
} else {
|
|
m_perspectiveCorrection->setEnabled( true );
|
|
}
|
|
}
|
|
|
|
void KisToolDuplicate::buttonPress(KisButtonPressEvent *e)
|
|
{
|
|
if (e->state() == ShiftButton) {
|
|
m_position = e->pos();
|
|
m_isOffsetNotUptodate = true;
|
|
} else {
|
|
if (m_position != TQPoint(-1, -1)) {
|
|
super::buttonPress(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void KisToolDuplicate::setup(KActionCollection *collection)
|
|
{
|
|
m_action = static_cast<KRadioAction *>(collection->action(name()));
|
|
|
|
if (m_action == 0) {
|
|
m_action = new KRadioAction(i18n("&Duplicate Brush"),
|
|
"tool_duplicate", TQt::Key_C, this,
|
|
TQT_SLOT(activate()), collection,
|
|
name());
|
|
m_action->setToolTip(i18n("Duplicate parts of the image. Shift-click to select the point to duplicate from to begin."));
|
|
m_action->setExclusiveGroup("tools");
|
|
m_ownAction = true;
|
|
}
|
|
}
|
|
|
|
void KisToolDuplicate::initPaint(KisEvent *e)
|
|
{
|
|
if( m_position != TQPoint(-1,-1))
|
|
{
|
|
if(m_isOffsetNotUptodate)
|
|
{
|
|
m_offset = e->pos() - m_position;
|
|
m_isOffsetNotUptodate = false;
|
|
}
|
|
m_paintIncremental = false;
|
|
super::initPaint(e);
|
|
painter()->setDuplicateOffset( m_offset );
|
|
KisPaintOp * op = KisPaintOpRegistry::instance()->paintOp("duplicate", 0, painter());
|
|
if (op && m_source) {
|
|
op->setSource(m_source);
|
|
painter()->setPaintOp(op);
|
|
}
|
|
m_positionStartPainting = e->pos();
|
|
painter()->setDuplicateStart( e->pos() );
|
|
}
|
|
}
|
|
|
|
void KisToolDuplicate::move(KisMoveEvent *e)
|
|
{
|
|
|
|
// Paint the outline where we will (or are) copying from
|
|
if( m_position == TQPoint(-1,-1) )
|
|
return;
|
|
|
|
TQPoint srcPos;
|
|
if (m_mode == PAINT) {
|
|
// if we are in perspective correction mode, update the offset when moving
|
|
if(m_perspectiveCorrection->isChecked())
|
|
{
|
|
double startM[3][3];
|
|
double endM[3][3];
|
|
for(int i = 0; i < 3; i++)
|
|
{
|
|
for(int j = 0; j < 3; j++)
|
|
{
|
|
startM[i][j] = 0.;
|
|
endM[i][j] = 0.;
|
|
}
|
|
startM[i][i] = 1.;
|
|
endM[i][i] = 1.;
|
|
}
|
|
|
|
// First look for the grid corresponding to the start point
|
|
KisSubPerspectiveGrid* subGridStart = *m_subject->currentImg()->perspectiveGrid()->begin();//device->image()->perspectiveGrid()->gridAt(KisPoint(srcPoint.x() +hotSpot.x(),srcPoint.y() +hotSpot.y()));
|
|
TQRect r = TQRect(0,0, m_subject->currentImg()->width(), m_subject->currentImg()->height());
|
|
|
|
if(subGridStart)
|
|
{
|
|
double* b = KisPerspectiveMath::computeMatrixTransfoFromPerspective( r, *subGridStart->topLeft(), *subGridStart->topRight(), *subGridStart->bottomLeft(), *subGridStart->bottomRight());
|
|
for(int i = 0; i < 3; i++)
|
|
{
|
|
for(int j = 0; j < 3; j++)
|
|
{
|
|
startM[i][j] = b[3*i+j];
|
|
}
|
|
}
|
|
|
|
}
|
|
// Second look for the grid corresponding to the end point
|
|
KisSubPerspectiveGrid* subGridEnd = *m_subject->currentImg()->perspectiveGrid()->begin();// device->image()->perspectiveGrid()->gridAt(pos);
|
|
if(subGridEnd)
|
|
{
|
|
double* b = KisPerspectiveMath::computeMatrixTransfoToPerspective(*subGridEnd->topLeft(), *subGridEnd->topRight(), *subGridEnd->bottomLeft(), *subGridEnd->bottomRight(), r);
|
|
for(int i = 0; i < 3; i++)
|
|
{
|
|
for(int j = 0; j < 3; j++)
|
|
{
|
|
endM[i][j] = b[3*i+j];
|
|
}
|
|
}
|
|
}
|
|
// Compute the translation in the perspective transformation space:
|
|
KisPoint translat;
|
|
{
|
|
KisPoint positionStartPaintingT = KisPerspectiveMath::matProd(endM, m_positionStartPainting);
|
|
KisPoint currentPositionT = KisPerspectiveMath::matProd(endM, e->pos() );
|
|
KisPoint duplicateStartPoisitionT = KisPerspectiveMath::matProd(endM, m_positionStartPainting - m_offset);
|
|
KisPoint duplicateRealPosition = KisPerspectiveMath::matProd(startM, duplicateStartPoisitionT + (currentPositionT - positionStartPaintingT) );
|
|
KisPoint p = e->pos() - duplicateRealPosition;
|
|
srcPos = p.floorTQPoint();
|
|
}
|
|
|
|
}else {
|
|
srcPos = painter()->duplicateOffset().floorTQPoint();
|
|
}
|
|
} else {
|
|
if(m_isOffsetNotUptodate)
|
|
srcPos = e->pos().floorTQPoint() - m_position.floorTQPoint();
|
|
else
|
|
srcPos = m_offset.floorTQPoint();
|
|
}
|
|
|
|
TQ_INT32 x;
|
|
TQ_INT32 y;
|
|
|
|
// like KisPaintOp::splitCoordinate
|
|
x = (TQ_INT32)((e->x() < 0) ? e->x() - 1 : e->x());
|
|
y = (TQ_INT32)((e->y() < 0) ? e->y() - 1 : e->y());
|
|
srcPos = TQPoint(x - srcPos.x(), y - srcPos.y());
|
|
|
|
paintOutline(srcPos);
|
|
super::move(e);
|
|
}
|
|
|
|
void KisToolDuplicate::paintAt(const KisPoint &pos,
|
|
const double pressure,
|
|
const double xtilt,
|
|
const double ytilt)
|
|
{
|
|
if( m_position != TQPoint(-1,-1))
|
|
{
|
|
if(m_isOffsetNotUptodate)
|
|
{
|
|
m_offset = pos - m_position;
|
|
m_isOffsetNotUptodate = false;
|
|
}
|
|
painter()->setDuplicateHealing( m_healing->isChecked() );
|
|
painter()->setDuplicateHealingRadius( m_healingRadius->value() );
|
|
painter()->setDuplicatePerspectiveCorrection( m_perspectiveCorrection->isChecked() );
|
|
painter()->paintAt( pos, pressure, xtilt, ytilt);
|
|
}
|
|
}
|
|
|
|
TQString KisToolDuplicate::quickHelp() const {
|
|
return i18n("To start, shift-click on the place you want to duplicate from. Then you can start painting. An indication of where you are copying from will be displayed while drawing and moving the mouse.");
|
|
}
|
|
|
|
TQWidget* KisToolDuplicate::createOptionWidget(TQWidget* tqparent)
|
|
{
|
|
TQWidget* widget = KisToolPaint::createOptionWidget(tqparent);
|
|
m_healing = new TQCheckBox(widget);
|
|
m_healing->setChecked( false);
|
|
addOptionWidgetOption(m_healing, new TQLabel(i18n("Healing"), widget ));
|
|
m_healingRadius = new KIntNumInput(widget);
|
|
|
|
KisBrush *brush = m_subject->currentBrush();
|
|
int healingradius = 20;
|
|
if( brush )
|
|
{
|
|
healingradius = 2 * TQMAX(brush->width(),brush->height());
|
|
}
|
|
|
|
m_healingRadius->setValue( healingradius );
|
|
addOptionWidgetOption(m_healingRadius, new TQLabel(i18n("Healing radius"), widget ));
|
|
m_perspectiveCorrection = new TQCheckBox(widget);
|
|
addOptionWidgetOption(m_perspectiveCorrection, new TQLabel(i18n("Correct the perspective"), widget ));
|
|
return widget;
|
|
}
|
|
|
|
#include "kis_tool_duplicate.moc"
|