// Copyright (C) 2002 Dominique Devriese // 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 "point_type.h" #include #include "special_imptypes.h" #include "point_imp.h" #include "curve_imp.h" #include "line_imp.h" #include "other_imp.h" #include "bogus_imp.h" #include "../modes/moving.h" #include "../misc/coordinate_system.h" #include "../misc/common.h" #include "../misc/calcpaths.h" #include "../misc/kiginputdialog.h" #include "../kig/kig_part.h" #include "../kig/kig_document.h" #include "../kig/kig_view.h" #include "../kig/kig_commands.h" #include static const ArgsParser::spec argsspecFixedPoint[] = { { DoubleImp::stype(), "x", "SHOULD NOT BE SEEN", false }, { DoubleImp::stype(), "y", "SHOULD NOT BE SEEN", false } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( FixedPointType ) FixedPointType::FixedPointType() : ArgsParserObjectType( "FixedPoint", argsspecFixedPoint, 2 ) { } FixedPointType::~FixedPointType() { } ObjectImp* FixedPointType::calc( const Args& parents, const KigDocument& ) const { if ( ! margsparser.checkArgs( parents ) ) return new InvalidImp; double a = static_cast( parents[0] )->data(); double b = static_cast( parents[1] )->data(); return new PointImp( Coordinate( a, b ) ); } static const ArgsParser::spec argsspecRelativePoint[] = { { DoubleImp::stype(), "relative-x", "SHOULD NOT BE SEEN", false }, { DoubleImp::stype(), "relative-y", "SHOULD NOT BE SEEN", false }, { ObjectImp::stype(), "object", "SHOULD NOT BE SEEN", false } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( RelativePointType ) RelativePointType::RelativePointType() : ArgsParserObjectType( "RelativePoint", argsspecRelativePoint, 3 ) { } RelativePointType::~RelativePointType() { } ObjectImp* RelativePointType::calc( const Args& parents, const KigDocument& ) const { if ( ! margsparser.checkArgs( parents ) ) return new InvalidImp; if ( ! parents[2]->attachPoint().valid() ) return new InvalidImp; Coordinate reference = static_cast( parents[2] )->attachPoint(); double a = static_cast( parents[0] )->data(); double b = static_cast( parents[1] )->data(); return new PointImp( reference + Coordinate( a, b ) ); } KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( CursorPointType ) CursorPointType::CursorPointType() : ObjectType( "CursorPoint" ) { } CursorPointType::~CursorPointType() { } const CursorPointType* CursorPointType::instance() { static const CursorPointType t; return &t; } ObjectImp* CursorPointType::calc( const Args& parents, const KigDocument& ) const { assert ( parents[0]->inherits( DoubleImp::stype() ) ); assert ( parents[1]->inherits( DoubleImp::stype() ) ); double a = static_cast( parents[0] )->data(); double b = static_cast( parents[1] )->data(); return new BogusPointImp( Coordinate( a, b ) ); } const ObjectImpType* CursorPointType::resultId() const { return BogusPointImp::stype(); } ObjectImp* ConstrainedPointType::calc( const Args& parents, const KigDocument& doc ) const { if ( ! margsparser.checkArgs( parents ) ) return new InvalidImp; double param = static_cast( parents[0] )->data(); const Coordinate nc = static_cast( parents[1] )->getPoint( param, doc ); doc.mcachedparam = param; if ( nc.valid() ) return new PointImp( nc ); else return new InvalidImp; } const ArgsParser::spec argsspecConstrainedPoint[] = { { DoubleImp::stype(), "parameter", "SHOULD NOT BE SEEN", false }, { CurveImp::stype(), "Constrain the point to this curve", "SHOULD NOT BE SEEN", true } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( ConstrainedPointType ) ConstrainedPointType::ConstrainedPointType() : ArgsParserObjectType( "ConstrainedPoint", argsspecConstrainedPoint, 2 ) { } ConstrainedPointType::~ConstrainedPointType() { } void FixedPointType::move( ObjectTypeCalcer& ourobj, const Coordinate& to, const KigDocument& ) const { // fetch the old coord..; std::vector pa = ourobj.parents(); assert( margsparser.checkArgs( pa ) ); assert( dynamic_cast( pa.front() ) ); assert( dynamic_cast( pa.back() ) ); ObjectConstCalcer* ox = static_cast( pa.front() ); ObjectConstCalcer* oy = static_cast( pa.back() ); ox->setImp( new DoubleImp( to.x ) ); oy->setImp( new DoubleImp( to.y ) ); } void RelativePointType::move( ObjectTypeCalcer& ourobj, const Coordinate& to, const KigDocument& ) const { // fetch the attach point..; // this routine is tightly paired with what moveReferencePoint returns! // right now moveReferencePoint always returns the origin std::vector pa = ourobj.parents(); assert( margsparser.checkArgs( pa ) ); assert( dynamic_cast( pa[0] ) ); assert( dynamic_cast( pa[1] ) ); ObjectConstCalcer* ox = static_cast( pa[0] ); ObjectConstCalcer* oy = static_cast( pa[1] ); ObjectCalcer* ob = static_cast( pa[2] ); Coordinate attach = ob->imp()->attachPoint(); ox->setImp( new DoubleImp( to.x - attach.x ) ); oy->setImp( new DoubleImp( to.y - attach.y ) ); } void CursorPointType::move( ObjectTypeCalcer& ourobj, const Coordinate& to, const KigDocument& ) const { // fetch the old coord..; std::vector pa = ourobj.parents(); assert( pa.size() == 2 ); assert( dynamic_cast( pa.front() ) ); assert( dynamic_cast( pa.back() ) ); ObjectConstCalcer* ox = static_cast( pa.front() ); ObjectConstCalcer* oy = static_cast( pa.back() ); ox->setImp( new DoubleImp( to.x ) ); oy->setImp( new DoubleImp( to.y ) ); } void ConstrainedPointType::move( ObjectTypeCalcer& ourobj, const Coordinate& to, const KigDocument& d ) const { // fetch the CurveImp.. std::vector parents = ourobj.parents(); assert( margsparser.checkArgs( parents ) ); assert( dynamic_cast( parents[0] ) ); ObjectConstCalcer* paramo = static_cast( parents[0] ); const CurveImp* ci = static_cast( parents[1]->imp() ); // fetch the new param.. const double np = ci->getParam( to, d ); paramo->setImp( new DoubleImp( np ) ); } bool ConstrainedPointType::canMove( const ObjectTypeCalcer& ) const { return true; } bool ConstrainedPointType::isFreelyTranslatable( const ObjectTypeCalcer& ) const { return false; } bool FixedPointType::canMove( const ObjectTypeCalcer& ) const { return true; } bool FixedPointType::isFreelyTranslatable( const ObjectTypeCalcer& ) const { return true; } bool RelativePointType::canMove( const ObjectTypeCalcer& ) const { return true; } bool RelativePointType::isFreelyTranslatable( const ObjectTypeCalcer& ) const { return true; } bool CursorPointType::canMove( const ObjectTypeCalcer& ) const { return true; } static const ArgsParser::spec argsspecMidPoint[] = { { PointImp::stype(), I18N_NOOP( "Construct the midpoint of this point and another point" ), I18N_NOOP( "Select the first of the two points of which you want to construct the midpoint..." ), false }, { PointImp::stype(), I18N_NOOP( "Construct the midpoint of this point and another point" ), I18N_NOOP( "Select the other of the two points of which you want to construct the midpoint..." ), false } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( MidPointType ) MidPointType::MidPointType() : ObjectABType( "MidPoint", argsspecMidPoint, 2 ) { } MidPointType::~MidPointType() { } const MidPointType* MidPointType::instance() { static const MidPointType t; return &t; } ObjectImp* MidPointType::calcx( const Coordinate& a, const Coordinate& b ) const { return new PointImp( ( a + b ) / 2 ); } bool ConstrainedPointType::inherits( int type ) const { return type == ID_ConstrainedPointType; } const ConstrainedPointType* ConstrainedPointType::instance() { static const ConstrainedPointType t; return &t; } bool FixedPointType::inherits( int type ) const { return type == ID_FixedPointType; } const FixedPointType* FixedPointType::instance() { static const FixedPointType t; return &t; } const ObjectImpType* FixedPointType::resultId() const { return PointImp::stype(); } const RelativePointType* RelativePointType::instance() { static const RelativePointType t; return &t; } const ObjectImpType* RelativePointType::resultId() const { return PointImp::stype(); } const ObjectImpType* ConstrainedPointType::resultId() const { return PointImp::stype(); } const ObjectImpType* CursorPointType::impRequirement( const ObjectImp* o, const Args& ) const { if ( o->inherits( DoubleImp::stype() ) ) return DoubleImp::stype(); if ( o->inherits( PointImp::stype() ) ) return PointImp::stype(); return 0; } bool CursorPointType::isDefinedOnOrThrough( const ObjectImp*, const Args& ) const { return false; } std::vector CursorPointType::sortArgs( const std::vector& args ) const { return args; } Args CursorPointType::sortArgs( const Args& args ) const { return args; } const ObjectImpType* MidPointType::resultId() const { return PointImp::stype(); } QStringList FixedPointType::specialActions() const { QStringList ret; ret << i18n( "Set &Coordinate..." ); ret << i18n( "Redefine" ); return ret; } QStringList ConstrainedPointType::specialActions() const { QStringList ret; ret << i18n( "Set &Parameter..." ); ret << i18n( "Redefine" ); return ret; } static void redefinePoint( ObjectHolder* o, KigPart& d, KigWidget& w ) { PointRedefineMode pm( o, d, w ); d.runMode( &pm ); } void FixedPointType::executeAction( int i, ObjectHolder& oh, ObjectTypeCalcer& o, KigPart& d, KigWidget& w, NormalMode& ) const { switch( i ) { case 0: { bool ok = true; assert ( o.imp()->inherits( PointImp::stype() ) ); Coordinate oldc = static_cast( o.imp() )->coordinate(); KigInputDialog::getCoordinate( i18n( "Set Coordinate" ), i18n( "Enter the new coordinate." ) + QString::fromLatin1( "
" ) + d.document().coordinateSystem().coordinateFormatNoticeMarkup(), &w, &ok, d.document(), &oldc ); if ( ! ok ) break; MonitorDataObjects mon( getAllParents( &o ) ); o.move( oldc, d.document() ); KigCommand* kc = new KigCommand( d, PointImp::stype()->moveAStatement() ); mon.finish( kc ); d.history()->push( kc ); break; }; case 1: redefinePoint( &oh, d, w ); break; default: assert( false ); }; } void ConstrainedPointType::executeAction( int i, ObjectHolder& oh, ObjectTypeCalcer& o, KigPart& d, KigWidget& w, NormalMode& ) const { switch( i ) { case 1: redefinePoint( &oh, d, w ); break; case 0: { std::vector parents = o.parents(); assert( dynamic_cast( parents[0] ) && parents[0]->imp()->inherits( DoubleImp::stype() ) ); ObjectConstCalcer* po = static_cast( parents[0] ); double oldp = static_cast( po->imp() )->data(); bool ok = true; double newp = getDoubleFromUser( i18n( "Set Point Parameter" ), i18n( "Choose the new parameter: " ), oldp, &w, &ok, 0, 1, 4 ); if ( ! ok ) return; MonitorDataObjects mon( parents ); po->setImp( new DoubleImp( newp ) ); KigCommand* kc = new KigCommand( d, i18n( "Change Parameter of Constrained Point" ) ); mon.finish( kc ); d.history()->push( kc ); break; }; default: assert( false ); }; } const Coordinate FixedPointType::moveReferencePoint( const ObjectTypeCalcer& ourobj ) const { assert( ourobj.imp()->inherits( PointImp::stype() ) ); return static_cast( ourobj.imp() )->coordinate(); } const Coordinate RelativePointType::moveReferencePoint( const ObjectTypeCalcer& ourobj ) const { assert( ourobj.imp()->inherits( PointImp::stype() ) ); // return static_cast( ourobj.imp() )->coordinate(); return Coordinate( 0., 0. ); } const Coordinate ConstrainedPointType::moveReferencePoint( const ObjectTypeCalcer& ourobj ) const { assert( ourobj.imp()->inherits( PointImp::stype() ) ); return static_cast( ourobj.imp() )->coordinate(); } std::vector FixedPointType::movableParents( const ObjectTypeCalcer& ourobj ) const { return ourobj.parents(); } std::vector RelativePointType::movableParents( const ObjectTypeCalcer& ourobj ) const { std::vector ret; ret.push_back( ourobj.parents()[0] ); ret.push_back( ourobj.parents()[1] ); return ret; } std::vector ConstrainedPointType::movableParents( const ObjectTypeCalcer& ourobj ) const { std::vector ret; ret.push_back( ourobj.parents()[0] ); return ret; } static const ArgsParser::spec argsspecConstrainedRelativePoint[] = { { DoubleImp::stype(), "relative-x", "SHOULD NOT BE SEEN", false }, { DoubleImp::stype(), "relative-y", "SHOULD NOT BE SEEN", false }, { DoubleImp::stype(), "parameter", "SHOULD NOT BE SEEN", false }, { CurveImp::stype(), "curve", "SHOULD NOT BE SEEN", false } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( ConstrainedRelativePointType ) ConstrainedRelativePointType::ConstrainedRelativePointType() : ArgsParserObjectType( "ConstrainedRelativePoint", argsspecConstrainedRelativePoint, 4 ) { } ConstrainedRelativePointType::~ConstrainedRelativePointType() { } ObjectImp* ConstrainedRelativePointType::calc( const Args& parents, const KigDocument& doc ) const { if ( ! margsparser.checkArgs( parents ) ) return new InvalidImp; double a = static_cast( parents[0] )->data(); double b = static_cast( parents[1] )->data(); double p = static_cast( parents[2] )->data(); Coordinate reference = static_cast( parents[3] )->getPoint( p, doc ); return new PointImp( reference + Coordinate( a, b ) ); } void ConstrainedRelativePointType::move( ObjectTypeCalcer& ourobj, const Coordinate& to, const KigDocument& doc ) const { // this routine is tightly paired with what moveReferencePoint returns! // right now moveReferencePoint always returns the origin std::vector pa = ourobj.parents(); assert( margsparser.checkArgs( pa ) ); assert( dynamic_cast( pa[0] ) ); assert( dynamic_cast( pa[1] ) ); assert( dynamic_cast( pa[2] ) ); ObjectConstCalcer* ox = static_cast( pa[0] ); ObjectConstCalcer* oy = static_cast( pa[1] ); ObjectConstCalcer* op = static_cast( pa[2] ); ObjectCalcer* ob = static_cast( pa[3] ); const CurveImp* curve = static_cast( ob->imp() ); double newp = curve->getParam( to, doc ); Coordinate attach = curve->getPoint( newp, doc ); ox->setImp( new DoubleImp( to.x - attach.x ) ); oy->setImp( new DoubleImp( to.y - attach.y ) ); op->setImp( new DoubleImp( newp ) ); } bool ConstrainedRelativePointType::canMove( const ObjectTypeCalcer& ) const { return true; } bool ConstrainedRelativePointType::isFreelyTranslatable( const ObjectTypeCalcer& ) const { return true; } const ConstrainedRelativePointType* ConstrainedRelativePointType::instance() { static const ConstrainedRelativePointType t; return &t; } const ObjectImpType* ConstrainedRelativePointType::resultId() const { return PointImp::stype(); } const Coordinate ConstrainedRelativePointType::moveReferencePoint( const ObjectTypeCalcer& ourobj ) const { assert( ourobj.imp()->inherits( PointImp::stype() ) ); return Coordinate( 0., 0. ); } std::vector ConstrainedRelativePointType::movableParents( const ObjectTypeCalcer& ourobj ) const { std::vector ret; ret.push_back( ourobj.parents()[0] ); ret.push_back( ourobj.parents()[1] ); ret.push_back( ourobj.parents()[2] ); return ret; } /* ----------------- Transport of measure ------------------------------ */ ObjectImp* MeasureTransportType::calc( const Args& parents, const KigDocument& doc ) const { double measure; if ( parents.size() != 3 ) return new InvalidImp; bool valid; measure = getDoubleFromImp( parents[0], valid ); if ( ! valid ) return new InvalidImp; const Coordinate& p = static_cast( parents[2] )->coordinate(); if ( parents[1]->inherits (LineImp::stype()) ) { const LineImp* c = static_cast( parents[1] ); if ( !c->containsPoint( p, doc ) ) return new InvalidImp; const LineData line = c->data(); const Coordinate dir = line.dir()/line.length(); const Coordinate nc = p + measure*dir; if ( nc.valid() ) return new PointImp( nc ); else return new InvalidImp; } else if ( parents[1]->inherits (CircleImp::stype()) ) { const CircleImp* c = static_cast( parents[1] ); if ( !c->containsPoint( p, doc ) ) return new InvalidImp; double param = c->getParam( p, doc ); measure /= 2*c->radius()*M_PI; param += measure; while (param > 1) param -= 1; const Coordinate nc = c->getPoint( param, doc ); if ( nc.valid() ) return new PointImp( nc ); else return new InvalidImp; } return new InvalidImp; } // I18N_NOOP( "Select the segment/arc to transport on the circle/line..." ), false }, // I18N_NOOP( "Select the circle/line on which to transport a measure..." ), true }, // I18N_NOOP( "Select a point on the circle/line..." ), false } KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( MeasureTransportType ) MeasureTransportType::MeasureTransportType() : ObjectType( "TransportOfMeasure" ) { } MeasureTransportType::~MeasureTransportType() { } const MeasureTransportType* MeasureTransportType::instance() { static const MeasureTransportType t; return &t; } const ObjectImpType* MeasureTransportType::resultId() const { return PointImp::stype(); } const ObjectImpType* MeasureTransportType::impRequirement( const ObjectImp* obj, const Args& ) const { if ( obj->inherits( PointImp::stype () ) ) return PointImp::stype (); if ( obj->inherits( LineImp::stype () ) ) return LineImp::stype (); if ( obj->inherits( CircleImp::stype () ) ) return CircleImp::stype (); if ( obj->inherits( SegmentImp::stype () ) ) return SegmentImp::stype (); if ( obj->inherits( ArcImp::stype () ) ) return ArcImp::stype (); return 0; } bool MeasureTransportType::isDefinedOnOrThrough( const ObjectImp* o, const Args& ) const { if ( o->inherits( LineImp::stype() ) ) return true; if ( o->inherits( CircleImp::stype() ) ) return true; return false; } std::vector MeasureTransportType::sortArgs( const std::vector& args ) const { return args; /* should already be in correct order */ } Args MeasureTransportType::sortArgs( const Args& args ) const { return args; } /* - transport of measure (old, for compatibility with prev. kig files) - */ /* it should be safe now to remove this completely, since the loading of "MeasureTransport" * is treated in "load07" building a node that uses the new version */ ObjectImp* MeasureTransportTypeOld::calc( const Args& parents, const KigDocument& doc ) const { if ( ! margsparser.checkArgs( parents ) ) return new InvalidImp; const CircleImp* c = static_cast( parents[0] ); const Coordinate& p = static_cast( parents[1] )->coordinate(); if ( !c->containsPoint( p, doc ) ) return new InvalidImp; const SegmentImp* s = static_cast( parents[2] ); double param = c->getParam( p, doc ); double measure = s->length(); measure /= 2*c->radius()*M_PI; param += measure; while (param > 1) param -= 1; const Coordinate nc = c->getPoint( param, doc ); if ( nc.valid() ) return new PointImp( nc ); else return new InvalidImp; } static const ArgsParser::spec argsspecMeasureTransportOld[] = { { CircleImp::stype(), "Transport a measure on this circle", I18N_NOOP( "Select the circle on which to transport a measure..." ), true }, { PointImp::stype(), "Start transport from this point of the circle", I18N_NOOP( "Select a point on the circle..." ), false }, { SegmentImp::stype(), "Segment to transport", I18N_NOOP( "Select the segment to transport on the circle..." ), false } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( MeasureTransportTypeOld ) MeasureTransportTypeOld::MeasureTransportTypeOld() : ArgsParserObjectType( "MeasureTransportObsoleted", argsspecMeasureTransportOld, 3 ) { } MeasureTransportTypeOld::~MeasureTransportTypeOld() { } const MeasureTransportTypeOld* MeasureTransportTypeOld::instance() { static const MeasureTransportTypeOld t; return &t; } const ObjectImpType* MeasureTransportTypeOld::resultId() const { return PointImp::stype(); } /* ----------------- end transport of measure ------------------------- */ /* Construct a point whose coordinates are indicated by numeric labels */ ObjectImp* PointByCoordsType::calc( const Args& parents, const KigDocument& ) const { if ( ! margsparser.checkArgs( parents ) ) return new InvalidImp; bool valid; double x = getDoubleFromImp( parents[0], valid); if ( ! valid ) return new InvalidImp; double y = getDoubleFromImp( parents[1], valid); if ( ! valid ) return new InvalidImp; const Coordinate nc = Coordinate( x, y ); if ( nc.valid() ) return new PointImp( nc ); else return new InvalidImp; } static const ArgsParser::spec argsspecPointByCoords[] = { { &lengthimptypeinstance, "X coordinate given by this number/length", I18N_NOOP( "Select a number/length as x coordinate of the point..." ), false }, { &lengthimptypeinstance, "Y coordinate given by this number/length", I18N_NOOP( "Select a number/length as y coordinate of the point..." ), false } }; KIG_INSTANTIATE_OBJECT_TYPE_INSTANCE( PointByCoordsType ) PointByCoordsType::PointByCoordsType() : ArgsParserObjectType( "PointByCoordinates", argsspecPointByCoords, 2 ) { } PointByCoordsType::~PointByCoordsType() { } const PointByCoordsType* PointByCoordsType::instance() { static const PointByCoordsType t; return &t; } const ObjectImpType* PointByCoordsType::resultId() const { return PointImp::stype(); }