|
|
|
/* This file is part of the KDE project
|
|
|
|
Copyright (C) 2001, 2002, 2003 The Karbon Developers
|
|
|
|
|
|
|
|
This library 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 library 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 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 <math.h>
|
|
|
|
|
|
|
|
#include <tqcursor.h>
|
|
|
|
#include <tqevent.h>
|
|
|
|
#include <tqlabel.h>
|
|
|
|
#include <tqgroupbox.h>
|
|
|
|
#include <tqcombobox.h>
|
|
|
|
#include <tqcheckbox.h>
|
|
|
|
#include <tqvbox.h>
|
|
|
|
#include <tqwidgetstack.h>
|
|
|
|
|
|
|
|
#include <klocale.h>
|
|
|
|
#include <knuminput.h>
|
|
|
|
//#include <KoUnitWidgets.h>
|
|
|
|
|
|
|
|
#include <karbon_part.h>
|
|
|
|
#include <karbon_view.h>
|
|
|
|
|
|
|
|
#include <core/vcolor.h>
|
|
|
|
#include <core/vcomposite.h>
|
|
|
|
#include <core/vfill.h>
|
|
|
|
#include <core/vstroke.h>
|
|
|
|
#include <core/vglobal.h>
|
|
|
|
#include <core/vselection.h>
|
|
|
|
#include <core/vcursor.h>
|
|
|
|
#include <render/vpainter.h>
|
|
|
|
#include <render/vpainterfactory.h>
|
|
|
|
#include "vpenciltool.h"
|
|
|
|
#include <commands/vtqshapecmd.h>
|
|
|
|
|
|
|
|
#include "vcurvefit.h"
|
|
|
|
|
|
|
|
#include "vpenciltool.moc"
|
|
|
|
|
|
|
|
VPencilOptionsWidget::VPencilOptionsWidget( KarbonView*view, TQWidget* tqparent, const char* name )
|
|
|
|
: KDialogBase( tqparent, name, true, i18n( "Pencil Settings" ), Ok | Cancel ), m_view( view )
|
|
|
|
{
|
|
|
|
TQVBox *vbox = new TQVBox( this );
|
|
|
|
|
|
|
|
m_combo = new TQComboBox( vbox );
|
|
|
|
|
|
|
|
m_combo->insertItem( i18n( "Raw" ) );
|
|
|
|
m_combo->insertItem( i18n( "Curve" ) );
|
|
|
|
m_combo->insertItem( i18n( "Straight" ) );
|
|
|
|
|
|
|
|
m_widgetStack = new TQWidgetStack( vbox );
|
|
|
|
|
|
|
|
TQGroupBox *group1 = new TQGroupBox( 2, Qt::Horizontal, i18n( "Properties" ), m_widgetStack );
|
|
|
|
m_widgetStack->addWidget( group1, 1 );
|
|
|
|
m_optimizeRaw = new TQCheckBox( i18n( "Optimize" ), group1 );
|
|
|
|
|
|
|
|
group1->setInsideMargin( 4 );
|
|
|
|
group1->setInsideSpacing( 2 );
|
|
|
|
|
|
|
|
TQGroupBox *group2 = new TQGroupBox( 2, Qt::Horizontal, i18n( "Properties" ), m_widgetStack );
|
|
|
|
m_widgetStack->addWidget( group2, 2 );
|
|
|
|
|
|
|
|
TQVBox *vbox2 = new TQVBox( group2 );
|
|
|
|
|
|
|
|
m_optimizeCurve = new TQCheckBox( i18n( "Optimize" ), vbox2 );
|
|
|
|
m_fittingError = new KDoubleNumInput( 0.0, 400.0, 4.00, 0.50, 3, vbox2 );
|
|
|
|
m_fittingError->setLabel( i18n( "Exactness:" ) );
|
|
|
|
|
|
|
|
group2->setInsideMargin( 4 );
|
|
|
|
group2->setInsideSpacing( 2 );
|
|
|
|
|
|
|
|
TQGroupBox *group3 = new TQGroupBox( 2, Qt::Horizontal, i18n( "Properties" ), m_widgetStack );
|
|
|
|
m_widgetStack->addWidget( group3, 3 );
|
|
|
|
|
|
|
|
m_combineAngle = new KDoubleNumInput( 0.0, 360.0, 0.10, 0.50, 3, group3 );
|
|
|
|
m_combineAngle->setSuffix( " deg" );
|
|
|
|
m_combineAngle->setLabel( i18n( "Combine angle:" ) );
|
|
|
|
|
|
|
|
group3->setInsideMargin( 4 );
|
|
|
|
group3->setInsideSpacing( 2 );
|
|
|
|
|
|
|
|
connect( m_combo, TQT_SIGNAL( activated( int ) ), this, TQT_SLOT( selectMode() ) );
|
|
|
|
|
|
|
|
//Set the default settings
|
|
|
|
m_mode = VPencilTool::CURVE;
|
|
|
|
selectMode();
|
|
|
|
|
|
|
|
m_optimizeCurve->setChecked( true );
|
|
|
|
m_optimizeRaw->setChecked( true );
|
|
|
|
|
|
|
|
setMainWidget( vbox );
|
|
|
|
}
|
|
|
|
|
|
|
|
float VPencilOptionsWidget::combineAngle()
|
|
|
|
{
|
|
|
|
return m_combineAngle->value();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VPencilOptionsWidget::optimize()
|
|
|
|
{
|
|
|
|
return ( m_optimizeRaw->isChecked() || m_optimizeCurve->isChecked() );
|
|
|
|
}
|
|
|
|
|
|
|
|
float VPencilOptionsWidget::fittingError()
|
|
|
|
{
|
|
|
|
return m_fittingError->value();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VPencilOptionsWidget::selectMode()
|
|
|
|
{
|
|
|
|
m_widgetStack->raiseWidget( m_combo->currentItem() + 1 );
|
|
|
|
|
|
|
|
switch( m_combo->currentItem() )
|
|
|
|
{
|
|
|
|
case 0: m_mode = VPencilTool::RAW; break;
|
|
|
|
case 1: m_mode = VPencilTool::CURVE; break;
|
|
|
|
case 2: m_mode = VPencilTool::STRAIGHT; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int VPencilOptionsWidget::currentMode(){
|
|
|
|
return m_mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
VPencilTool::VPencilTool( KarbonView *view )
|
|
|
|
: VTool( view, "tool_pencil" )
|
|
|
|
{
|
|
|
|
m_Points.setAutoDelete( true );
|
|
|
|
m_optionWidget = new VPencilOptionsWidget( view );
|
|
|
|
registerTool( this );
|
|
|
|
m_mode = CURVE;
|
|
|
|
m_optimize = true;
|
|
|
|
m_combineAngle = 3.0f;
|
|
|
|
m_cursor = new TQCursor( VCursor::createCursor( VCursor::CrossHair ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
VPencilTool::~VPencilTool()
|
|
|
|
{
|
|
|
|
delete m_cursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
TQString
|
|
|
|
VPencilTool::contextHelp()
|
|
|
|
{
|
|
|
|
TQString s = i18n( "<qt><b>Pencil tool:</b><br>" );
|
|
|
|
s += i18n( "- <i>Click</i> to begin drawing, release when you have finished.");
|
|
|
|
s += i18n( "- Press <i>Enter</i> or <i>double click</i> to end the polyline.</qt>" );
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::activate()
|
|
|
|
{
|
|
|
|
VTool::activate();
|
|
|
|
view()->statusMessage()->setText( i18n( "Pencil Tool" ) );
|
|
|
|
view()->setCursor( *m_cursor );
|
|
|
|
view()->part()->document().selection()->showHandle( false );
|
|
|
|
|
|
|
|
m_Points.clear();
|
|
|
|
m_close = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::deactivate()
|
|
|
|
{
|
|
|
|
m_Points.removeLast();
|
|
|
|
m_Points.removeLast();
|
|
|
|
|
|
|
|
VPath* line = 0L;
|
|
|
|
|
|
|
|
TQPtrList<KoPoint> complete;
|
|
|
|
TQPtrList<KoPoint> *points = &m_Points;
|
|
|
|
|
|
|
|
if( m_Points.count() > 1 )
|
|
|
|
{
|
|
|
|
if( m_optimize || m_mode == STRAIGHT )
|
|
|
|
{
|
|
|
|
complete.setAutoDelete( true );
|
|
|
|
m_Points.setAutoDelete( false );
|
|
|
|
|
|
|
|
float cangle;
|
|
|
|
|
|
|
|
if( m_mode == STRAIGHT )
|
|
|
|
cangle = m_combineAngle;
|
|
|
|
else
|
|
|
|
cangle = 0.50f;
|
|
|
|
|
|
|
|
#define ANGLE(P0,P1)\
|
|
|
|
atan((P1)->y()-(P0)->y())/((P1)->x()-(P0)->x())*(180/M_PI)
|
|
|
|
|
|
|
|
//Add the first point
|
|
|
|
complete.append( m_Points.first() );
|
|
|
|
complete.append( m_Points.next() );
|
|
|
|
|
|
|
|
//Now we need to get the angle of the first line
|
|
|
|
float langle = ANGLE( complete.at( 0 ), complete.at( 1 ) );
|
|
|
|
|
|
|
|
KoPoint *nextp = NULL;
|
|
|
|
while( ( nextp = m_Points.next() ) )
|
|
|
|
{
|
|
|
|
float angle = ANGLE( complete.last(), nextp );
|
|
|
|
if( TQABS( angle - langle ) < cangle )
|
|
|
|
complete.removeLast();
|
|
|
|
complete.append(nextp);
|
|
|
|
langle=angle;
|
|
|
|
}
|
|
|
|
m_Points.clear();
|
|
|
|
m_Points.setAutoDelete(true);
|
|
|
|
|
|
|
|
points = &complete;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch(m_mode)
|
|
|
|
{
|
|
|
|
case CURVE:
|
|
|
|
{
|
|
|
|
line = bezierFit( *points, m_optionWidget->fittingError() );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case STRAIGHT:
|
|
|
|
case RAW:
|
|
|
|
{
|
|
|
|
line = new VPath( 0L );
|
|
|
|
KoPoint* p1 = (*points).first();
|
|
|
|
KoPoint* plast = p1;
|
|
|
|
line->moveTo( *p1 );
|
|
|
|
|
|
|
|
KoPoint* pnext = 0L;
|
|
|
|
|
|
|
|
while( ( pnext = (*points).next() ) )
|
|
|
|
{
|
|
|
|
line->lineTo( *pnext );
|
|
|
|
plast = pnext;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( shiftPressed() )
|
|
|
|
line->close();
|
|
|
|
}
|
|
|
|
|
|
|
|
if( line )
|
|
|
|
{
|
|
|
|
VShapeCmd* cmd = new VShapeCmd(
|
|
|
|
&view()->part()->document(),
|
|
|
|
i18n( "Pencil" ),
|
|
|
|
line,
|
|
|
|
"14_pencil" );
|
|
|
|
|
|
|
|
view()->part()->addCommand( cmd, true );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::draw()
|
|
|
|
{
|
|
|
|
VPainter* painter = view()->painterFactory()->editpainter();
|
|
|
|
painter->setRasterOp( TQt::NotROP );
|
|
|
|
|
|
|
|
m_mode = m_optionWidget->currentMode();
|
|
|
|
m_optimize = m_optionWidget->optimize();
|
|
|
|
m_combineAngle = m_optionWidget->combineAngle();
|
|
|
|
|
|
|
|
if( m_Points.count() > 1 )
|
|
|
|
{
|
|
|
|
VPath line( 0L );
|
|
|
|
line.moveTo( *m_Points.first() );
|
|
|
|
|
|
|
|
KoPoint *pnext;
|
|
|
|
while((pnext=m_Points.next())){
|
|
|
|
line.lineTo( *pnext );
|
|
|
|
}
|
|
|
|
|
|
|
|
line.setState( VObject::edit );
|
|
|
|
line.draw( painter, &line.boundingBox() );
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseMove()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseButtonPress()
|
|
|
|
{
|
|
|
|
m_Points.append( new KoPoint( last() ) );
|
|
|
|
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseButtonRelease()
|
|
|
|
{
|
|
|
|
m_Points.append( new KoPoint( last() ) );
|
|
|
|
draw();
|
|
|
|
accept();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseButtonDblClick()
|
|
|
|
{
|
|
|
|
accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseDrag()
|
|
|
|
{
|
|
|
|
if( m_Points.count() != 0 )
|
|
|
|
{
|
|
|
|
draw();
|
|
|
|
|
|
|
|
m_Points.append( new KoPoint( last() ) );
|
|
|
|
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseDragRelease()
|
|
|
|
{
|
|
|
|
mouseButtonRelease();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseDragShiftPressed()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseDragCtrlPressed()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseDragShiftReleased()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::mouseDragCtrlReleased()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::cancel()
|
|
|
|
{
|
|
|
|
draw();
|
|
|
|
|
|
|
|
m_Points.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::cancelStep()
|
|
|
|
{
|
|
|
|
draw();
|
|
|
|
|
|
|
|
m_Points.clear();
|
|
|
|
|
|
|
|
draw();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::accept()
|
|
|
|
{
|
|
|
|
deactivate();
|
|
|
|
activate();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
VPencilTool::showDialog() const
|
|
|
|
{
|
|
|
|
return m_optionWidget->exec() == TQDialog::Accepted;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
VPencilTool::setup( KActionCollection *collection )
|
|
|
|
{
|
|
|
|
m_action = static_cast<KRadioAction *>(collection -> action( name() ) );
|
|
|
|
|
|
|
|
if( m_action == 0 )
|
|
|
|
{
|
|
|
|
m_action = new KRadioAction( i18n( "Pencil Tool" ), "14_pencil", TQt::SHIFT+TQt::Key_P, this, TQT_SLOT( activate() ), collection, name() );
|
|
|
|
m_action->setToolTip( i18n( "Pencil" ) );
|
|
|
|
m_action->setExclusiveGroup( "freehand" );
|
|
|
|
//m_ownAction = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|