/* * kis_tool_duplicate.cpp - part of Chalk * * Copyright (c) 2004 Cyrille Berger * * 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 #include #include #include #include #include #include #include #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(TDEActionCollection *collection) { m_action = static_cast(collection->action(name())); if (m_action == 0) { m_action = new TDERadioAction(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* parent) { TQWidget* widget = KisToolPaint::createOptionWidget(parent); 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"