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.

1737 lines
58 KiB

  1. /*
  2. * This program source code file is part of KiCad, a free EDA CAD application.
  3. *
  4. * Copyright (C) 2012 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
  5. * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
  6. * Copyright (C) 2012 Wayne Stambaugh <stambaughw@gmail.com>
  7. * Copyright (C) 2023 CERN
  8. * Copyright (C) 1992-2023 KiCad Developers, see AUTHORS.txt for contributors.
  9. *
  10. * This program is free software; you can redistribute it and/or
  11. * modify it under the terms of the GNU General Public License
  12. * as published by the Free Software Foundation; either version 2
  13. * of the License, or (at your option) any later version.
  14. *
  15. * This program is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. * GNU General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU General Public License
  21. * along with this program; if not, you may find one here:
  22. * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
  23. * or you may search the http://www.gnu.org website for the version 2 license,
  24. * or you may write to the Free Software Foundation, Inc.,
  25. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
  26. */
  27. #include <bitmaps.h>
  28. #include <pcb_edit_frame.h>
  29. #include <base_units.h>
  30. #include <convert_basic_shapes_to_polygon.h>
  31. #include <font/font.h>
  32. #include <board.h>
  33. #include <pcb_dimension.h>
  34. #include <pcb_text.h>
  35. #include <geometry/shape_compound.h>
  36. #include <geometry/shape_circle.h>
  37. #include <geometry/shape_segment.h>
  38. #include <settings/color_settings.h>
  39. #include <settings/settings_manager.h>
  40. #include <trigo.h>
  41. static const EDA_ANGLE s_arrowAngle( 27.5, DEGREES_T );
  42. PCB_DIMENSION_BASE::PCB_DIMENSION_BASE( BOARD_ITEM* aParent, KICAD_T aType ) :
  43. PCB_TEXT( aParent, aType ),
  44. m_overrideTextEnabled( false ),
  45. m_units( EDA_UNITS::INCHES ),
  46. m_autoUnits( false ),
  47. m_unitsFormat( DIM_UNITS_FORMAT::BARE_SUFFIX ),
  48. m_precision( DIM_PRECISION::X_XXXX ),
  49. m_suppressZeroes( false ),
  50. m_lineThickness( pcbIUScale.mmToIU( 0.2 ) ),
  51. m_arrowLength( pcbIUScale.MilsToIU( 50 ) ),
  52. m_extensionOffset( 0 ),
  53. m_textPosition( DIM_TEXT_POSITION::OUTSIDE ),
  54. m_keepTextAligned( true ),
  55. m_measuredValue( 0 ),
  56. m_inClearRenderCache( false )
  57. {
  58. m_layer = Dwgs_User;
  59. }
  60. bool PCB_DIMENSION_BASE::operator==( const BOARD_ITEM& aOther ) const
  61. {
  62. if( Type() != aOther.Type() )
  63. return false;
  64. const PCB_DIMENSION_BASE& other = static_cast<const PCB_DIMENSION_BASE&>( aOther );
  65. return *this == other;
  66. }
  67. bool PCB_DIMENSION_BASE::operator==( const PCB_DIMENSION_BASE& aOther ) const
  68. {
  69. if( m_textPosition != aOther.m_textPosition )
  70. return false;
  71. if( m_keepTextAligned != aOther.m_keepTextAligned )
  72. return false;
  73. if( m_units != aOther.m_units )
  74. return false;
  75. if( m_autoUnits != aOther.m_autoUnits )
  76. return false;
  77. if( m_unitsFormat != aOther.m_unitsFormat )
  78. return false;
  79. if( m_precision != aOther.m_precision )
  80. return false;
  81. if( m_suppressZeroes != aOther.m_suppressZeroes )
  82. return false;
  83. if( m_lineThickness != aOther.m_lineThickness )
  84. return false;
  85. if( m_arrowLength != aOther.m_arrowLength )
  86. return false;
  87. if( m_extensionOffset != aOther.m_extensionOffset )
  88. return false;
  89. if( m_measuredValue != aOther.m_measuredValue )
  90. return false;
  91. return EDA_TEXT::operator==( aOther );
  92. }
  93. double PCB_DIMENSION_BASE::Similarity( const BOARD_ITEM& aOther ) const
  94. {
  95. if( m_Uuid == aOther.m_Uuid )
  96. return 1.0;
  97. if( Type() != aOther.Type() )
  98. return 0.0;
  99. const PCB_DIMENSION_BASE& other = static_cast<const PCB_DIMENSION_BASE&>( aOther );
  100. double similarity = 1.0;
  101. if( m_textPosition != other.m_textPosition )
  102. similarity *= 0.9;
  103. if( m_keepTextAligned != other.m_keepTextAligned )
  104. similarity *= 0.9;
  105. if( m_units != other.m_units )
  106. similarity *= 0.9;
  107. if( m_autoUnits != other.m_autoUnits )
  108. similarity *= 0.9;
  109. if( m_unitsFormat != other.m_unitsFormat )
  110. similarity *= 0.9;
  111. if( m_precision != other.m_precision )
  112. similarity *= 0.9;
  113. if( m_suppressZeroes != other.m_suppressZeroes )
  114. similarity *= 0.9;
  115. if( m_lineThickness != other.m_lineThickness )
  116. similarity *= 0.9;
  117. if( m_arrowLength != other.m_arrowLength )
  118. similarity *= 0.9;
  119. if( m_extensionOffset != other.m_extensionOffset )
  120. similarity *= 0.9;
  121. if( m_measuredValue != other.m_measuredValue )
  122. similarity *= 0.9;
  123. similarity *= EDA_TEXT::Similarity( other );
  124. return similarity;
  125. }
  126. void PCB_DIMENSION_BASE::updateText()
  127. {
  128. wxString text = m_overrideTextEnabled ? m_valueString : GetValueText();
  129. switch( m_unitsFormat )
  130. {
  131. case DIM_UNITS_FORMAT::NO_SUFFIX: // no units
  132. break;
  133. case DIM_UNITS_FORMAT::BARE_SUFFIX: // normal
  134. text += EDA_UNIT_UTILS::GetText( m_units );
  135. break;
  136. case DIM_UNITS_FORMAT::PAREN_SUFFIX: // parenthetical
  137. text += wxT( " (" ) + EDA_UNIT_UTILS::GetText( m_units ).Trim( false ) + wxT( ")" );
  138. break;
  139. }
  140. text.Prepend( m_prefix );
  141. text.Append( m_suffix );
  142. SetText( text );
  143. }
  144. void PCB_DIMENSION_BASE::ClearRenderCache()
  145. {
  146. PCB_TEXT::ClearRenderCache();
  147. // We use EDA_TEXT::ClearRenderCache() as a signal that the properties of the EDA_TEXT
  148. // have changed and we may need to update the dimension text
  149. if( !m_inClearRenderCache )
  150. {
  151. m_inClearRenderCache = true;
  152. Update();
  153. m_inClearRenderCache = false;
  154. }
  155. }
  156. template<typename ShapeType>
  157. void PCB_DIMENSION_BASE::addShape( const ShapeType& aShape )
  158. {
  159. m_shapes.push_back( std::make_shared<ShapeType>( aShape ) );
  160. }
  161. wxString PCB_DIMENSION_BASE::GetValueText() const
  162. {
  163. struct lconv* lc = localeconv();
  164. wxChar sep = lc->decimal_point[0];
  165. int val = GetMeasuredValue();
  166. int precision = static_cast<int>( m_precision );
  167. wxString text;
  168. if( precision >= 6 )
  169. {
  170. switch( m_units )
  171. {
  172. case EDA_UNITS::INCHES: precision = precision - 4; break;
  173. case EDA_UNITS::MILS: precision = std::max( 0, precision - 7 ); break;
  174. case EDA_UNITS::MILLIMETRES: precision = precision - 5; break;
  175. default: precision = precision - 4; break;
  176. }
  177. }
  178. wxString format = wxT( "%." ) + wxString::Format( wxT( "%i" ), precision ) + wxT( "f" );
  179. text.Printf( format, EDA_UNIT_UTILS::UI::ToUserUnit( pcbIUScale, m_units, val ) );
  180. if( m_suppressZeroes )
  181. {
  182. while( text.Last() == '0' )
  183. {
  184. text.RemoveLast();
  185. if( text.Last() == '.' || text.Last() == sep )
  186. {
  187. text.RemoveLast();
  188. break;
  189. }
  190. }
  191. }
  192. return text;
  193. }
  194. void PCB_DIMENSION_BASE::SetPrefix( const wxString& aPrefix )
  195. {
  196. m_prefix = aPrefix;
  197. }
  198. void PCB_DIMENSION_BASE::SetSuffix( const wxString& aSuffix )
  199. {
  200. m_suffix = aSuffix;
  201. }
  202. void PCB_DIMENSION_BASE::SetUnits( EDA_UNITS aUnits )
  203. {
  204. m_units = aUnits;
  205. }
  206. DIM_UNITS_MODE PCB_DIMENSION_BASE::GetUnitsMode() const
  207. {
  208. if( m_autoUnits )
  209. {
  210. return DIM_UNITS_MODE::AUTOMATIC;
  211. }
  212. else
  213. {
  214. switch( m_units )
  215. {
  216. default:
  217. case EDA_UNITS::INCHES: return DIM_UNITS_MODE::INCHES;
  218. case EDA_UNITS::MILLIMETRES: return DIM_UNITS_MODE::MILLIMETRES;
  219. case EDA_UNITS::MILS: return DIM_UNITS_MODE::MILS;
  220. }
  221. }
  222. }
  223. void PCB_DIMENSION_BASE::SetUnitsMode( DIM_UNITS_MODE aMode )
  224. {
  225. switch( aMode )
  226. {
  227. case DIM_UNITS_MODE::INCHES:
  228. m_autoUnits = false;
  229. m_units = EDA_UNITS::INCHES;
  230. break;
  231. case DIM_UNITS_MODE::MILS:
  232. m_autoUnits = false;
  233. m_units = EDA_UNITS::MILS;
  234. break;
  235. case DIM_UNITS_MODE::MILLIMETRES:
  236. m_autoUnits = false;
  237. m_units = EDA_UNITS::MILLIMETRES;
  238. break;
  239. case DIM_UNITS_MODE::AUTOMATIC:
  240. m_autoUnits = true;
  241. m_units = GetBoard() ? GetBoard()->GetUserUnits() : EDA_UNITS::MILLIMETRES;
  242. break;
  243. }
  244. }
  245. void PCB_DIMENSION_BASE::ChangeTextAngleDegrees( double aDegrees )
  246. {
  247. SetTextAngleDegrees( aDegrees );
  248. // Create or repair any knockouts
  249. Update();
  250. }
  251. void PCB_DIMENSION_BASE::ChangeKeepTextAligned( bool aKeepAligned )
  252. {
  253. SetKeepTextAligned( aKeepAligned );
  254. // Re-align the text and repair any knockouts
  255. Update();
  256. }
  257. void PCB_DIMENSION_BASE::Move( const VECTOR2I& offset )
  258. {
  259. PCB_TEXT::Offset( offset );
  260. m_start += offset;
  261. m_end += offset;
  262. Update();
  263. }
  264. void PCB_DIMENSION_BASE::Rotate( const VECTOR2I& aRotCentre, const EDA_ANGLE& aAngle )
  265. {
  266. EDA_ANGLE newAngle = GetTextAngle() + aAngle;
  267. newAngle.Normalize();
  268. SetTextAngle( newAngle );
  269. VECTOR2I pt = GetTextPos();
  270. RotatePoint( pt, aRotCentre, aAngle );
  271. SetTextPos( pt );
  272. RotatePoint( m_start, aRotCentre, aAngle );
  273. RotatePoint( m_end, aRotCentre, aAngle );
  274. Update();
  275. }
  276. void PCB_DIMENSION_BASE::Flip( const VECTOR2I& aCentre, FLIP_DIRECTION aFlipDirection )
  277. {
  278. Mirror( aCentre, aFlipDirection );
  279. SetLayer( GetBoard()->FlipLayer( GetLayer() ) );
  280. }
  281. void PCB_DIMENSION_BASE::Mirror( const VECTOR2I& axis_pos, FLIP_DIRECTION aFlipDirection )
  282. {
  283. VECTOR2I newPos = GetTextPos();
  284. MIRROR( newPos, axis_pos, aFlipDirection );
  285. SetTextPos( newPos );
  286. // invert angle
  287. SetTextAngle( -GetTextAngle() );
  288. MIRROR( m_start, axis_pos, aFlipDirection );
  289. MIRROR( m_end, axis_pos, aFlipDirection );
  290. if( IsSideSpecific() )
  291. SetMirrored( !IsMirrored() );
  292. Update();
  293. }
  294. void PCB_DIMENSION_BASE::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame,
  295. std::vector<MSG_PANEL_ITEM>& aList )
  296. {
  297. // for now, display only the text within the DIMENSION using class PCB_TEXT.
  298. wxString msg;
  299. wxCHECK_RET( m_parent != nullptr, wxT( "PCB_TEXT::GetMsgPanelInfo() m_Parent is NULL." ) );
  300. // Don't use GetShownText(); we want to see the variable references here
  301. aList.emplace_back( _( "Dimension" ), KIUI::EllipsizeStatusText( aFrame, GetText() ) );
  302. aList.emplace_back( _( "Prefix" ), GetPrefix() );
  303. if( GetOverrideTextEnabled() )
  304. {
  305. aList.emplace_back( _( "Override Text" ), GetOverrideText() );
  306. }
  307. else
  308. {
  309. aList.emplace_back( _( "Value" ), GetValueText() );
  310. switch( GetPrecision() )
  311. {
  312. case DIM_PRECISION::V_VV: msg = wxT( "0.00 in / 0 mils / 0.0 mm" ); break;
  313. case DIM_PRECISION::V_VVV: msg = wxT( "0.000 in / 0 mils / 0.00 mm" ); break;
  314. case DIM_PRECISION::V_VVVV: msg = wxT( "0.0000 in / 0.0 mils / 0.000 mm" ); break;
  315. case DIM_PRECISION::V_VVVVV: msg = wxT( "0.00000 in / 0.00 mils / 0.0000 mm" ); break;
  316. default: msg = wxT( "%" ) + wxString::Format( wxT( "1.%df" ), GetPrecision() );
  317. }
  318. aList.emplace_back( _( "Precision" ), wxString::Format( msg, 0.0 ) );
  319. }
  320. aList.emplace_back( _( "Suffix" ), GetSuffix() );
  321. // Use our own UNITS_PROVIDER to report dimension info in dimension's units rather than
  322. // in frame's units.
  323. UNITS_PROVIDER unitsProvider( pcbIUScale, EDA_UNITS::MILLIMETRES );
  324. unitsProvider.SetUserUnits( GetUnits() );
  325. aList.emplace_back( _( "Units" ), EDA_UNIT_UTILS::GetLabel( GetUnits() ) );
  326. aList.emplace_back( _( "Font" ), GetFont() ? GetFont()->GetName() : _( "Default" ) );
  327. aList.emplace_back( _( "Text Thickness" ), unitsProvider.MessageTextFromValue( GetTextThickness() ) );
  328. aList.emplace_back( _( "Text Width" ), unitsProvider.MessageTextFromValue( GetTextWidth() ) );
  329. aList.emplace_back( _( "Text Height" ), unitsProvider.MessageTextFromValue( GetTextHeight() ) );
  330. ORIGIN_TRANSFORMS& originTransforms = aFrame->GetOriginTransforms();
  331. if( Type() == PCB_DIM_CENTER_T )
  332. {
  333. VECTOR2I startCoord = originTransforms.ToDisplayAbs( GetStart() );
  334. wxString start = wxString::Format( wxT( "@(%s, %s)" ),
  335. aFrame->MessageTextFromValue( startCoord.x ),
  336. aFrame->MessageTextFromValue( startCoord.y ) );
  337. aList.emplace_back( start, wxEmptyString );
  338. }
  339. else
  340. {
  341. VECTOR2I startCoord = originTransforms.ToDisplayAbs( GetStart() );
  342. wxString start = wxString::Format( wxT( "@(%s, %s)" ),
  343. aFrame->MessageTextFromValue( startCoord.x ),
  344. aFrame->MessageTextFromValue( startCoord.y ) );
  345. VECTOR2I endCoord = originTransforms.ToDisplayAbs( GetEnd() );
  346. wxString end = wxString::Format( wxT( "@(%s, %s)" ),
  347. aFrame->MessageTextFromValue( endCoord.x ),
  348. aFrame->MessageTextFromValue( endCoord.y ) );
  349. aList.emplace_back( start, end );
  350. }
  351. if( aFrame->GetName() == PCB_EDIT_FRAME_NAME && IsLocked() )
  352. aList.emplace_back( _( "Status" ), _( "Locked" ) );
  353. aList.emplace_back( _( "Layer" ), GetLayerName() );
  354. }
  355. std::shared_ptr<SHAPE> PCB_DIMENSION_BASE::GetEffectiveShape( PCB_LAYER_ID aLayer, FLASHING aFlash ) const
  356. {
  357. std::shared_ptr<SHAPE_COMPOUND> effectiveShape = std::make_shared<SHAPE_COMPOUND>();
  358. effectiveShape->AddShape( GetEffectiveTextShape()->Clone() );
  359. for( const std::shared_ptr<SHAPE>& shape : GetShapes() )
  360. effectiveShape->AddShape( shape->Clone() );
  361. return effectiveShape;
  362. }
  363. bool PCB_DIMENSION_BASE::HitTest( const VECTOR2I& aPosition, int aAccuracy ) const
  364. {
  365. if( TextHitTest( aPosition ) )
  366. return true;
  367. int dist_max = aAccuracy + ( m_lineThickness / 2 );
  368. // Locate SEGMENTS
  369. for( const std::shared_ptr<SHAPE>& shape : GetShapes() )
  370. {
  371. if( shape->Collide( aPosition, dist_max ) )
  372. return true;
  373. }
  374. return false;
  375. }
  376. bool PCB_DIMENSION_BASE::HitTest( const BOX2I& aRect, bool aContained, int aAccuracy ) const
  377. {
  378. BOX2I arect = aRect;
  379. arect.Inflate( aAccuracy );
  380. BOX2I rect = GetBoundingBox();
  381. if( aAccuracy )
  382. rect.Inflate( aAccuracy );
  383. if( aContained )
  384. return arect.Contains( rect );
  385. return arect.Intersects( rect );
  386. }
  387. const BOX2I PCB_DIMENSION_BASE::GetBoundingBox() const
  388. {
  389. BOX2I bBox;
  390. int xmin, xmax, ymin, ymax;
  391. bBox = GetTextBox();
  392. xmin = bBox.GetX();
  393. xmax = bBox.GetRight();
  394. ymin = bBox.GetY();
  395. ymax = bBox.GetBottom();
  396. for( const std::shared_ptr<SHAPE>& shape : GetShapes() )
  397. {
  398. BOX2I shapeBox = shape->BBox();
  399. shapeBox.Inflate( m_lineThickness / 2 );
  400. xmin = std::min( xmin, shapeBox.GetOrigin().x );
  401. xmax = std::max( xmax, shapeBox.GetEnd().x );
  402. ymin = std::min( ymin, shapeBox.GetOrigin().y );
  403. ymax = std::max( ymax, shapeBox.GetEnd().y );
  404. }
  405. bBox.SetX( xmin );
  406. bBox.SetY( ymin );
  407. bBox.SetWidth( xmax - xmin + 1 );
  408. bBox.SetHeight( ymax - ymin + 1 );
  409. bBox.Normalize();
  410. return bBox;
  411. }
  412. wxString PCB_DIMENSION_BASE::GetItemDescription( UNITS_PROVIDER* aUnitsProvider, bool aFull ) const
  413. {
  414. return wxString::Format( _( "Dimension '%s' on %s" ),
  415. aFull ? GetShownText( false ) : KIUI::EllipsizeMenuText( GetText() ),
  416. GetLayerName() );
  417. }
  418. const BOX2I PCB_DIMENSION_BASE::ViewBBox() const
  419. {
  420. BOX2I dimBBox = BOX2I( VECTOR2I( GetBoundingBox().GetPosition() ),
  421. VECTOR2I( GetBoundingBox().GetSize() ) );
  422. dimBBox.Merge( PCB_TEXT::ViewBBox() );
  423. return dimBBox;
  424. }
  425. OPT_VECTOR2I PCB_DIMENSION_BASE::segPolyIntersection( const SHAPE_POLY_SET& aPoly, const SEG& aSeg,
  426. bool aStart )
  427. {
  428. VECTOR2I start( aStart ? aSeg.A : aSeg.B );
  429. VECTOR2I endpoint( aStart ? aSeg.B : aSeg.A );
  430. if( aPoly.Contains( start ) )
  431. return std::nullopt;
  432. for( SHAPE_POLY_SET::CONST_SEGMENT_ITERATOR seg = aPoly.CIterateSegments(); seg; ++seg )
  433. {
  434. if( OPT_VECTOR2I intersection = ( *seg ).Intersect( aSeg ) )
  435. {
  436. if( ( *intersection - start ).SquaredEuclideanNorm() <
  437. ( endpoint - start ).SquaredEuclideanNorm() )
  438. endpoint = *intersection;
  439. }
  440. }
  441. if( start == endpoint )
  442. return std::nullopt;
  443. return OPT_VECTOR2I( endpoint );
  444. }
  445. OPT_VECTOR2I PCB_DIMENSION_BASE::segCircleIntersection( CIRCLE& aCircle, SEG& aSeg, bool aStart )
  446. {
  447. VECTOR2I start( aStart ? aSeg.A : aSeg.B );
  448. VECTOR2I endpoint( aStart ? aSeg.B : aSeg.A );
  449. if( aCircle.Contains( start ) )
  450. return std::nullopt;
  451. std::vector<VECTOR2I> intersections = aCircle.Intersect( aSeg );
  452. for( VECTOR2I& intersection : aCircle.Intersect( aSeg ) )
  453. {
  454. if( ( intersection - start ).SquaredEuclideanNorm() <
  455. ( endpoint - start ).SquaredEuclideanNorm() )
  456. endpoint = intersection;
  457. }
  458. if( start == endpoint )
  459. return std::nullopt;
  460. return OPT_VECTOR2I( endpoint );
  461. }
  462. void PCB_DIMENSION_BASE::TransformShapeToPolygon( SHAPE_POLY_SET& aBuffer, PCB_LAYER_ID aLayer,
  463. int aClearance, int aError, ERROR_LOC aErrorLoc,
  464. bool aIgnoreLineWidth ) const
  465. {
  466. wxASSERT_MSG( !aIgnoreLineWidth, wxT( "IgnoreLineWidth has no meaning for dimensions." ) );
  467. for( const std::shared_ptr<SHAPE>& shape : m_shapes )
  468. {
  469. const SHAPE_CIRCLE* circle = dynamic_cast<const SHAPE_CIRCLE*>( shape.get() );
  470. const SHAPE_SEGMENT* seg = dynamic_cast<const SHAPE_SEGMENT*>( shape.get() );
  471. if( circle )
  472. {
  473. TransformCircleToPolygon( aBuffer, circle->GetCenter(),
  474. circle->GetRadius() + m_lineThickness / 2 + aClearance,
  475. aError, aErrorLoc );
  476. }
  477. else if( seg )
  478. {
  479. TransformOvalToPolygon( aBuffer, seg->GetSeg().A, seg->GetSeg().B,
  480. m_lineThickness + 2 * aClearance, aError, aErrorLoc );
  481. }
  482. else
  483. {
  484. wxFAIL_MSG( wxT( "PCB_DIMENSION_BASE::TransformShapeToPolygon unknown shape type." ) );
  485. }
  486. }
  487. }
  488. PCB_DIM_ALIGNED::PCB_DIM_ALIGNED( BOARD_ITEM* aParent, KICAD_T aType ) :
  489. PCB_DIMENSION_BASE( aParent, aType ),
  490. m_height( 0 )
  491. {
  492. // To preserve look of old dimensions, initialize extension height based on default arrow length
  493. m_extensionHeight = static_cast<int>( m_arrowLength * s_arrowAngle.Sin() );
  494. }
  495. EDA_ITEM* PCB_DIM_ALIGNED::Clone() const
  496. {
  497. return new PCB_DIM_ALIGNED( *this );
  498. }
  499. void PCB_DIM_ALIGNED::swapData( BOARD_ITEM* aImage )
  500. {
  501. wxASSERT( aImage->Type() == Type() );
  502. m_shapes.clear();
  503. static_cast<PCB_DIM_ALIGNED*>( aImage )->m_shapes.clear();
  504. std::swap( *static_cast<PCB_DIM_ALIGNED*>( this ), *static_cast<PCB_DIM_ALIGNED*>( aImage ) );
  505. Update();
  506. }
  507. void PCB_DIM_ALIGNED::Mirror( const VECTOR2I& axis_pos, FLIP_DIRECTION aFlipDirection )
  508. {
  509. m_height = -m_height;
  510. // Call this last for the Update()
  511. PCB_DIMENSION_BASE::Mirror( axis_pos, aFlipDirection );
  512. }
  513. BITMAPS PCB_DIM_ALIGNED::GetMenuImage() const
  514. {
  515. return BITMAPS::add_aligned_dimension;
  516. }
  517. void PCB_DIM_ALIGNED::UpdateHeight( const VECTOR2I& aCrossbarStart, const VECTOR2I& aCrossbarEnd )
  518. {
  519. VECTOR2D height( aCrossbarStart - GetStart() );
  520. VECTOR2D crossBar( aCrossbarEnd - aCrossbarStart );
  521. if( height.Cross( crossBar ) > 0 )
  522. m_height = -height.EuclideanNorm();
  523. else
  524. m_height = height.EuclideanNorm();
  525. Update();
  526. }
  527. void PCB_DIM_ALIGNED::updateGeometry()
  528. {
  529. m_shapes.clear();
  530. VECTOR2I dimension( m_end - m_start );
  531. m_measuredValue = KiROUND( dimension.EuclideanNorm() );
  532. VECTOR2I extension;
  533. if( m_height > 0 )
  534. extension = VECTOR2I( -dimension.y, dimension.x );
  535. else
  536. extension = VECTOR2I( dimension.y, -dimension.x );
  537. // Add extension lines
  538. int extensionHeight = std::abs( m_height ) - m_extensionOffset + m_extensionHeight;
  539. VECTOR2I extStart( m_start );
  540. extStart += extension.Resize( m_extensionOffset );
  541. addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
  542. extStart = VECTOR2I( m_end );
  543. extStart += extension.Resize( m_extensionOffset );
  544. addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
  545. // Add crossbar
  546. VECTOR2I crossBarDistance = sign( m_height ) * extension.Resize( m_height );
  547. m_crossBarStart = m_start + crossBarDistance;
  548. m_crossBarEnd = m_end + crossBarDistance;
  549. // Update text after calculating crossbar position but before adding crossbar lines
  550. updateText();
  551. // Now that we have the text updated, we can determine how to draw the crossbar.
  552. // First we need to create an appropriate bounding polygon to collide with
  553. BOX2I textBox = GetTextBox().Inflate( GetTextWidth() / 2, - GetEffectiveTextPenWidth() );
  554. SHAPE_POLY_SET polyBox;
  555. polyBox.NewOutline();
  556. polyBox.Append( textBox.GetOrigin() );
  557. polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
  558. polyBox.Append( textBox.GetEnd() );
  559. polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
  560. polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
  561. // The ideal crossbar, if the text doesn't collide
  562. SEG crossbar( m_crossBarStart, m_crossBarEnd );
  563. // Now we can draw 0, 1, or 2 crossbar lines depending on how the polygon collides
  564. bool containsA = polyBox.Contains( crossbar.A );
  565. bool containsB = polyBox.Contains( crossbar.B );
  566. OPT_VECTOR2I endpointA = segPolyIntersection( polyBox, crossbar );
  567. OPT_VECTOR2I endpointB = segPolyIntersection( polyBox, crossbar, false );
  568. if( endpointA )
  569. m_shapes.emplace_back( new SHAPE_SEGMENT( crossbar.A, *endpointA ) );
  570. if( endpointB )
  571. m_shapes.emplace_back( new SHAPE_SEGMENT( *endpointB, crossbar.B ) );
  572. if( !containsA && !containsB && !endpointA && !endpointB )
  573. m_shapes.emplace_back( new SHAPE_SEGMENT( crossbar ) );
  574. // Add arrows
  575. VECTOR2I arrowEndPos( m_arrowLength, 0 );
  576. VECTOR2I arrowEndNeg( m_arrowLength, 0 );
  577. RotatePoint( arrowEndPos, -EDA_ANGLE( dimension ) + s_arrowAngle );
  578. RotatePoint( arrowEndNeg, -EDA_ANGLE( dimension ) - s_arrowAngle );
  579. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarStart, m_crossBarStart + arrowEndPos ) );
  580. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarStart, m_crossBarStart + arrowEndNeg ) );
  581. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarEnd, m_crossBarEnd - arrowEndPos ) );
  582. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarEnd, m_crossBarEnd - arrowEndNeg ) );
  583. }
  584. void PCB_DIM_ALIGNED::updateText()
  585. {
  586. VECTOR2I crossbarCenter( ( m_crossBarEnd - m_crossBarStart ) / 2 );
  587. if( m_textPosition == DIM_TEXT_POSITION::OUTSIDE )
  588. {
  589. int textOffsetDistance = GetEffectiveTextPenWidth() + GetTextHeight();
  590. EDA_ANGLE rotation;
  591. if( crossbarCenter.x == 0 )
  592. rotation = ANGLE_90 * sign( -crossbarCenter.y );
  593. else if( crossbarCenter.x < 0 )
  594. rotation = -ANGLE_90;
  595. else
  596. rotation = ANGLE_90;
  597. VECTOR2I textOffset = crossbarCenter;
  598. RotatePoint( textOffset, rotation );
  599. textOffset = crossbarCenter + textOffset.Resize( textOffsetDistance );
  600. SetTextPos( m_crossBarStart + textOffset );
  601. }
  602. else if( m_textPosition == DIM_TEXT_POSITION::INLINE )
  603. {
  604. SetTextPos( m_crossBarStart + crossbarCenter );
  605. }
  606. if( m_keepTextAligned )
  607. {
  608. EDA_ANGLE textAngle = FULL_CIRCLE - EDA_ANGLE( crossbarCenter );
  609. textAngle.Normalize();
  610. if( textAngle > ANGLE_90 && textAngle <= ANGLE_270 )
  611. textAngle -= ANGLE_180;
  612. SetTextAngle( textAngle );
  613. }
  614. PCB_DIMENSION_BASE::updateText();
  615. }
  616. void PCB_DIM_ALIGNED::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
  617. {
  618. PCB_DIMENSION_BASE::GetMsgPanelInfo( aFrame, aList );
  619. // Use our own UNITS_PROVIDER to report dimension info in dimension's units rather than
  620. // in frame's units.
  621. UNITS_PROVIDER unitsProvider( pcbIUScale, EDA_UNITS::MILLIMETRES );
  622. unitsProvider.SetUserUnits( GetUnits() );
  623. aList.emplace_back( _( "Height" ), unitsProvider.MessageTextFromValue( m_height ) );
  624. }
  625. PCB_DIM_ORTHOGONAL::PCB_DIM_ORTHOGONAL( BOARD_ITEM* aParent ) :
  626. PCB_DIM_ALIGNED( aParent, PCB_DIM_ORTHOGONAL_T )
  627. {
  628. // To preserve look of old dimensions, initialize extension height based on default arrow length
  629. m_extensionHeight = static_cast<int>( m_arrowLength * s_arrowAngle.Sin() );
  630. m_orientation = DIR::HORIZONTAL;
  631. }
  632. EDA_ITEM* PCB_DIM_ORTHOGONAL::Clone() const
  633. {
  634. return new PCB_DIM_ORTHOGONAL( *this );
  635. }
  636. void PCB_DIM_ORTHOGONAL::swapData( BOARD_ITEM* aImage )
  637. {
  638. wxASSERT( aImage->Type() == Type() );
  639. m_shapes.clear();
  640. static_cast<PCB_DIM_ORTHOGONAL*>( aImage )->m_shapes.clear();
  641. std::swap( *static_cast<PCB_DIM_ORTHOGONAL*>( this ),
  642. *static_cast<PCB_DIM_ORTHOGONAL*>( aImage ) );
  643. Update();
  644. }
  645. void PCB_DIM_ORTHOGONAL::Mirror( const VECTOR2I& axis_pos, FLIP_DIRECTION aFlipDirection )
  646. {
  647. // Only reverse the height if the height is aligned with the flip
  648. if( m_orientation == DIR::HORIZONTAL && aFlipDirection == FLIP_DIRECTION::TOP_BOTTOM )
  649. m_height = -m_height;
  650. else if( m_orientation == DIR::VERTICAL && aFlipDirection == FLIP_DIRECTION::LEFT_RIGHT )
  651. m_height = -m_height;
  652. // Call this last, as we need the Update()
  653. PCB_DIMENSION_BASE::Mirror( axis_pos, aFlipDirection );
  654. }
  655. BITMAPS PCB_DIM_ORTHOGONAL::GetMenuImage() const
  656. {
  657. return BITMAPS::add_orthogonal_dimension;
  658. }
  659. void PCB_DIM_ORTHOGONAL::updateGeometry()
  660. {
  661. m_shapes.clear();
  662. int measurement = ( m_orientation == DIR::HORIZONTAL ? m_end.x - m_start.x :
  663. m_end.y - m_start.y );
  664. m_measuredValue = KiROUND( std::abs( measurement ) );
  665. VECTOR2I extension;
  666. if( m_orientation == DIR::HORIZONTAL )
  667. extension = VECTOR2I( 0, m_height );
  668. else
  669. extension = VECTOR2I( m_height, 0 );
  670. // Add first extension line
  671. int extensionHeight = std::abs( m_height ) - m_extensionOffset + m_extensionHeight;
  672. VECTOR2I extStart( m_start );
  673. extStart += extension.Resize( m_extensionOffset );
  674. addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
  675. // Add crossbar
  676. VECTOR2I crossBarDistance = sign( m_height ) * extension.Resize( m_height );
  677. m_crossBarStart = m_start + crossBarDistance;
  678. if( m_orientation == DIR::HORIZONTAL )
  679. m_crossBarEnd = VECTOR2I( m_end.x, m_crossBarStart.y );
  680. else
  681. m_crossBarEnd = VECTOR2I( m_crossBarStart.x, m_end.y );
  682. // Add second extension line (m_end to crossbar end)
  683. if( m_orientation == DIR::HORIZONTAL )
  684. extension = VECTOR2I( 0, m_end.y - m_crossBarEnd.y );
  685. else
  686. extension = VECTOR2I( m_end.x - m_crossBarEnd.x, 0 );
  687. extensionHeight = extension.EuclideanNorm() - m_extensionOffset + m_extensionHeight;
  688. extStart = VECTOR2I( m_crossBarEnd );
  689. extStart -= extension.Resize( m_extensionHeight );
  690. addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
  691. // Update text after calculating crossbar position but before adding crossbar lines
  692. updateText();
  693. // Now that we have the text updated, we can determine how to draw the crossbar.
  694. // First we need to create an appropriate bounding polygon to collide with
  695. BOX2I textBox = GetTextBox().Inflate( GetTextWidth() / 2, GetEffectiveTextPenWidth() );
  696. SHAPE_POLY_SET polyBox;
  697. polyBox.NewOutline();
  698. polyBox.Append( textBox.GetOrigin() );
  699. polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
  700. polyBox.Append( textBox.GetEnd() );
  701. polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
  702. polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
  703. // The ideal crossbar, if the text doesn't collide
  704. SEG crossbar( m_crossBarStart, m_crossBarEnd );
  705. // Now we can draw 0, 1, or 2 crossbar lines depending on how the polygon collides
  706. bool containsA = polyBox.Contains( crossbar.A );
  707. bool containsB = polyBox.Contains( crossbar.B );
  708. OPT_VECTOR2I endpointA = segPolyIntersection( polyBox, crossbar );
  709. OPT_VECTOR2I endpointB = segPolyIntersection( polyBox, crossbar, false );
  710. if( endpointA )
  711. m_shapes.emplace_back( new SHAPE_SEGMENT( crossbar.A, *endpointA ) );
  712. if( endpointB )
  713. m_shapes.emplace_back( new SHAPE_SEGMENT( *endpointB, crossbar.B ) );
  714. if( !containsA && !containsB && !endpointA && !endpointB )
  715. m_shapes.emplace_back( new SHAPE_SEGMENT( crossbar ) );
  716. // Add arrows
  717. EDA_ANGLE crossBarAngle( m_crossBarEnd - m_crossBarStart );
  718. VECTOR2I arrowEndPos( m_arrowLength, 0 );
  719. VECTOR2I arrowEndNeg( m_arrowLength, 0 );
  720. RotatePoint( arrowEndPos, -crossBarAngle + s_arrowAngle );
  721. RotatePoint( arrowEndNeg, -crossBarAngle - s_arrowAngle );
  722. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarStart, m_crossBarStart + arrowEndPos ) );
  723. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarStart, m_crossBarStart + arrowEndNeg ) );
  724. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarEnd, m_crossBarEnd - arrowEndPos ) );
  725. m_shapes.emplace_back( new SHAPE_SEGMENT( m_crossBarEnd, m_crossBarEnd - arrowEndNeg ) );
  726. }
  727. void PCB_DIM_ORTHOGONAL::updateText()
  728. {
  729. VECTOR2I crossbarCenter( ( m_crossBarEnd - m_crossBarStart ) / 2 );
  730. if( m_textPosition == DIM_TEXT_POSITION::OUTSIDE )
  731. {
  732. int textOffsetDistance = GetEffectiveTextPenWidth() + GetTextHeight();
  733. VECTOR2I textOffset;
  734. if( m_orientation == DIR::HORIZONTAL )
  735. textOffset.y = -textOffsetDistance;
  736. else
  737. textOffset.x = -textOffsetDistance;
  738. textOffset += crossbarCenter;
  739. SetTextPos( m_crossBarStart + textOffset );
  740. }
  741. else if( m_textPosition == DIM_TEXT_POSITION::INLINE )
  742. {
  743. SetTextPos( m_crossBarStart + crossbarCenter );
  744. }
  745. if( m_keepTextAligned )
  746. {
  747. if( abs( crossbarCenter.x ) > abs( crossbarCenter.y ) )
  748. SetTextAngle( ANGLE_HORIZONTAL );
  749. else
  750. SetTextAngle( ANGLE_VERTICAL );
  751. }
  752. PCB_DIM_ALIGNED::updateText();
  753. }
  754. void PCB_DIM_ORTHOGONAL::Rotate( const VECTOR2I& aRotCentre, const EDA_ANGLE& aAngle )
  755. {
  756. EDA_ANGLE angle( aAngle );
  757. // restrict angle to -179.9 to 180.0 degrees
  758. angle.Normalize180();
  759. // adjust orientation and height to new angle
  760. // we can only handle the cases of -90, 0, 90, 180 degrees exactly;
  761. // in the other cases we will use the nearest 90 degree angle to
  762. // choose at least an approximate axis for the target orientation
  763. // In case of exactly 45 or 135 degrees, we will round towards zero for consistency
  764. if( angle > ANGLE_45 && angle <= ANGLE_135 )
  765. {
  766. // about 90 degree
  767. if( m_orientation == DIR::HORIZONTAL )
  768. {
  769. m_orientation = DIR::VERTICAL;
  770. }
  771. else
  772. {
  773. m_orientation = DIR::HORIZONTAL;
  774. m_height = -m_height;
  775. }
  776. }
  777. else if( angle < -ANGLE_45 && angle >= -ANGLE_135 )
  778. {
  779. // about -90 degree
  780. if( m_orientation == DIR::HORIZONTAL )
  781. {
  782. m_orientation = DIR::VERTICAL;
  783. m_height = -m_height;
  784. }
  785. else
  786. {
  787. m_orientation = DIR::HORIZONTAL;
  788. }
  789. }
  790. else if( angle > ANGLE_135 || angle < -ANGLE_135 )
  791. {
  792. // about 180 degree
  793. m_height = -m_height;
  794. }
  795. // this will update m_crossBarStart and m_crossbarEnd
  796. PCB_DIMENSION_BASE::Rotate( aRotCentre, angle );
  797. }
  798. PCB_DIM_LEADER::PCB_DIM_LEADER( BOARD_ITEM* aParent ) :
  799. PCB_DIMENSION_BASE( aParent, PCB_DIM_LEADER_T ),
  800. m_textBorder( DIM_TEXT_BORDER::NONE )
  801. {
  802. m_unitsFormat = DIM_UNITS_FORMAT::NO_SUFFIX;
  803. m_overrideTextEnabled = true;
  804. m_keepTextAligned = false;
  805. SetOverrideText( _( "Leader" ) );
  806. }
  807. EDA_ITEM* PCB_DIM_LEADER::Clone() const
  808. {
  809. return new PCB_DIM_LEADER( *this );
  810. }
  811. void PCB_DIM_LEADER::swapData( BOARD_ITEM* aImage )
  812. {
  813. wxASSERT( aImage->Type() == Type() );
  814. m_shapes.clear();
  815. static_cast<PCB_DIM_LEADER*>( aImage )->m_shapes.clear();
  816. std::swap( *static_cast<PCB_DIM_LEADER*>( this ), *static_cast<PCB_DIM_LEADER*>( aImage ) );
  817. Update();
  818. }
  819. BITMAPS PCB_DIM_LEADER::GetMenuImage() const
  820. {
  821. return BITMAPS::add_leader;
  822. }
  823. void PCB_DIM_LEADER::updateText()
  824. {
  825. // Our geometry is dependent on the size of the text, so just update the whole shebang
  826. updateGeometry();
  827. }
  828. void PCB_DIM_LEADER::updateGeometry()
  829. {
  830. m_shapes.clear();
  831. PCB_DIMENSION_BASE::updateText();
  832. // Now that we have the text updated, we can determine how to draw the second line
  833. // First we need to create an appropriate bounding polygon to collide with
  834. BOX2I textBox = GetTextBox().Inflate( GetTextWidth() / 2, GetEffectiveTextPenWidth() * 2 );
  835. SHAPE_POLY_SET polyBox;
  836. polyBox.NewOutline();
  837. polyBox.Append( textBox.GetOrigin() );
  838. polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
  839. polyBox.Append( textBox.GetEnd() );
  840. polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
  841. polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
  842. VECTOR2I firstLine( m_end - m_start );
  843. VECTOR2I start( m_start );
  844. start += firstLine.Resize( m_extensionOffset );
  845. SEG arrowSeg( m_start, m_end );
  846. SEG textSeg( m_end, GetTextPos() );
  847. OPT_VECTOR2I arrowSegEnd;
  848. OPT_VECTOR2I textSegEnd;
  849. if( m_textBorder == DIM_TEXT_BORDER::CIRCLE )
  850. {
  851. double penWidth = GetEffectiveTextPenWidth() / 2.0;
  852. double radius = ( textBox.GetWidth() / 2.0 ) - penWidth;
  853. CIRCLE circle( textBox.GetCenter(), radius );
  854. arrowSegEnd = segCircleIntersection( circle, arrowSeg );
  855. textSegEnd = segCircleIntersection( circle, textSeg );
  856. }
  857. else
  858. {
  859. arrowSegEnd = segPolyIntersection( polyBox, arrowSeg );
  860. textSegEnd = segPolyIntersection( polyBox, textSeg );
  861. }
  862. if( !arrowSegEnd )
  863. arrowSegEnd = m_end;
  864. m_shapes.emplace_back( new SHAPE_SEGMENT( start, *arrowSegEnd ) );
  865. // Add arrows
  866. VECTOR2I arrowEndPos( m_arrowLength, 0 );
  867. VECTOR2I arrowEndNeg( m_arrowLength, 0 );
  868. RotatePoint( arrowEndPos, -EDA_ANGLE( firstLine ) + s_arrowAngle );
  869. RotatePoint( arrowEndNeg, -EDA_ANGLE( firstLine ) - s_arrowAngle );
  870. m_shapes.emplace_back( new SHAPE_SEGMENT( start, start + arrowEndPos ) );
  871. m_shapes.emplace_back( new SHAPE_SEGMENT( start, start + arrowEndNeg ) );
  872. if( !GetText().IsEmpty() )
  873. {
  874. switch( m_textBorder )
  875. {
  876. case DIM_TEXT_BORDER::RECTANGLE:
  877. {
  878. for( SHAPE_POLY_SET::SEGMENT_ITERATOR seg = polyBox.IterateSegments(); seg; seg++ )
  879. m_shapes.emplace_back( new SHAPE_SEGMENT( *seg ) );
  880. break;
  881. }
  882. case DIM_TEXT_BORDER::CIRCLE:
  883. {
  884. double penWidth = GetEffectiveTextPenWidth() / 2.0;
  885. double radius = ( textBox.GetWidth() / 2.0 ) - penWidth;
  886. m_shapes.emplace_back( new SHAPE_CIRCLE( textBox.GetCenter(), radius ) );
  887. break;
  888. }
  889. default:
  890. break;
  891. }
  892. }
  893. if( textSegEnd && *arrowSegEnd == m_end )
  894. m_shapes.emplace_back( new SHAPE_SEGMENT( m_end, *textSegEnd ) );
  895. }
  896. void PCB_DIM_LEADER::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
  897. {
  898. // Don't use GetShownText(); we want to see the variable references here
  899. aList.emplace_back( _( "Leader" ), KIUI::EllipsizeStatusText( aFrame, GetText() ) );
  900. ORIGIN_TRANSFORMS& originTransforms = aFrame->GetOriginTransforms();
  901. VECTOR2I startCoord = originTransforms.ToDisplayAbs( GetStart() );
  902. wxString start = wxString::Format( wxT( "@(%s, %s)" ),
  903. aFrame->MessageTextFromValue( startCoord.x ),
  904. aFrame->MessageTextFromValue( startCoord.y ) );
  905. aList.emplace_back( start, wxEmptyString );
  906. aList.emplace_back( _( "Layer" ), GetLayerName() );
  907. }
  908. PCB_DIM_RADIAL::PCB_DIM_RADIAL( BOARD_ITEM* aParent ) :
  909. PCB_DIMENSION_BASE( aParent, PCB_DIM_RADIAL_T )
  910. {
  911. m_unitsFormat = DIM_UNITS_FORMAT::NO_SUFFIX;
  912. m_overrideTextEnabled = false;
  913. m_keepTextAligned = true;
  914. m_isDiameter = false;
  915. m_prefix = "R ";
  916. m_leaderLength = m_arrowLength * 3;
  917. }
  918. EDA_ITEM* PCB_DIM_RADIAL::Clone() const
  919. {
  920. return new PCB_DIM_RADIAL( *this );
  921. }
  922. void PCB_DIM_RADIAL::swapData( BOARD_ITEM* aImage )
  923. {
  924. wxASSERT( aImage->Type() == Type() );
  925. m_shapes.clear();
  926. static_cast<PCB_DIM_RADIAL*>( aImage )->m_shapes.clear();
  927. std::swap( *static_cast<PCB_DIM_RADIAL*>( this ), *static_cast<PCB_DIM_RADIAL*>( aImage ) );
  928. Update();
  929. }
  930. BITMAPS PCB_DIM_RADIAL::GetMenuImage() const
  931. {
  932. return BITMAPS::add_radial_dimension;
  933. }
  934. VECTOR2I PCB_DIM_RADIAL::GetKnee() const
  935. {
  936. VECTOR2I radial( m_end - m_start );
  937. return m_end + radial.Resize( m_leaderLength );
  938. }
  939. void PCB_DIM_RADIAL::updateText()
  940. {
  941. if( m_keepTextAligned )
  942. {
  943. VECTOR2I textLine( GetTextPos() - GetKnee() );
  944. EDA_ANGLE textAngle = FULL_CIRCLE - EDA_ANGLE( textLine );
  945. textAngle.Normalize();
  946. if( textAngle > ANGLE_90 && textAngle <= ANGLE_270 )
  947. textAngle -= ANGLE_180;
  948. // Round to nearest degree
  949. textAngle = EDA_ANGLE( KiROUND( textAngle.AsDegrees() ), DEGREES_T );
  950. SetTextAngle( textAngle );
  951. }
  952. PCB_DIMENSION_BASE::updateText();
  953. }
  954. void PCB_DIM_RADIAL::updateGeometry()
  955. {
  956. m_shapes.clear();
  957. VECTOR2I center( m_start );
  958. VECTOR2I centerArm( 0, m_arrowLength );
  959. m_shapes.emplace_back( new SHAPE_SEGMENT( center - centerArm, center + centerArm ) );
  960. RotatePoint( centerArm, -ANGLE_90 );
  961. m_shapes.emplace_back( new SHAPE_SEGMENT( center - centerArm, center + centerArm ) );
  962. VECTOR2I radius( m_end - m_start );
  963. if( m_isDiameter )
  964. m_measuredValue = KiROUND( radius.EuclideanNorm() * 2 );
  965. else
  966. m_measuredValue = KiROUND( radius.EuclideanNorm() );
  967. updateText();
  968. // Now that we have the text updated, we can determine how to draw the second line
  969. // First we need to create an appropriate bounding polygon to collide with
  970. BOX2I textBox = GetTextBox().Inflate( GetTextWidth() / 2, GetEffectiveTextPenWidth() );
  971. SHAPE_POLY_SET polyBox;
  972. polyBox.NewOutline();
  973. polyBox.Append( textBox.GetOrigin() );
  974. polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
  975. polyBox.Append( textBox.GetEnd() );
  976. polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
  977. polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
  978. VECTOR2I radial( m_end - m_start );
  979. radial = radial.Resize( m_leaderLength );
  980. SEG arrowSeg( m_end, m_end + radial );
  981. SEG textSeg( arrowSeg.B, GetTextPos() );
  982. OPT_VECTOR2I arrowSegEnd = segPolyIntersection( polyBox, arrowSeg );
  983. OPT_VECTOR2I textSegEnd = segPolyIntersection( polyBox, textSeg );
  984. if( arrowSegEnd )
  985. arrowSeg.B = *arrowSegEnd;
  986. if( textSegEnd )
  987. textSeg.B = *textSegEnd;
  988. m_shapes.emplace_back( new SHAPE_SEGMENT( arrowSeg ) );
  989. // Add arrows
  990. VECTOR2I arrowEndPos( m_arrowLength, 0 );
  991. VECTOR2I arrowEndNeg( m_arrowLength, 0 );
  992. RotatePoint( arrowEndPos, -EDA_ANGLE( radial ) + s_arrowAngle );
  993. RotatePoint( arrowEndNeg, -EDA_ANGLE( radial ) - s_arrowAngle );
  994. m_shapes.emplace_back( new SHAPE_SEGMENT( m_end, m_end + arrowEndPos ) );
  995. m_shapes.emplace_back( new SHAPE_SEGMENT( m_end, m_end + arrowEndNeg ) );
  996. m_shapes.emplace_back( new SHAPE_SEGMENT( textSeg ) );
  997. }
  998. PCB_DIM_CENTER::PCB_DIM_CENTER( BOARD_ITEM* aParent ) :
  999. PCB_DIMENSION_BASE( aParent, PCB_DIM_CENTER_T )
  1000. {
  1001. m_unitsFormat = DIM_UNITS_FORMAT::NO_SUFFIX;
  1002. m_overrideTextEnabled = true;
  1003. }
  1004. EDA_ITEM* PCB_DIM_CENTER::Clone() const
  1005. {
  1006. return new PCB_DIM_CENTER( *this );
  1007. }
  1008. void PCB_DIM_CENTER::swapData( BOARD_ITEM* aImage )
  1009. {
  1010. wxASSERT( aImage->Type() == Type() );
  1011. std::swap( *static_cast<PCB_DIM_CENTER*>( this ), *static_cast<PCB_DIM_CENTER*>( aImage ) );
  1012. }
  1013. BITMAPS PCB_DIM_CENTER::GetMenuImage() const
  1014. {
  1015. return BITMAPS::add_center_dimension;
  1016. }
  1017. const BOX2I PCB_DIM_CENTER::GetBoundingBox() const
  1018. {
  1019. int halfWidth = VECTOR2I( m_end - m_start ).x + ( m_lineThickness / 2.0 );
  1020. BOX2I bBox;
  1021. bBox.SetX( m_start.x - halfWidth );
  1022. bBox.SetY( m_start.y - halfWidth );
  1023. bBox.SetWidth( halfWidth * 2 );
  1024. bBox.SetHeight( halfWidth * 2 );
  1025. bBox.Normalize();
  1026. return bBox;
  1027. }
  1028. const BOX2I PCB_DIM_CENTER::ViewBBox() const
  1029. {
  1030. return BOX2I( VECTOR2I( GetBoundingBox().GetPosition() ),
  1031. VECTOR2I( GetBoundingBox().GetSize() ) );
  1032. }
  1033. void PCB_DIM_CENTER::updateGeometry()
  1034. {
  1035. m_shapes.clear();
  1036. VECTOR2I center( m_start );
  1037. VECTOR2I arm( m_end - m_start );
  1038. m_shapes.emplace_back( new SHAPE_SEGMENT( center - arm, center + arm ) );
  1039. RotatePoint( arm, -ANGLE_90 );
  1040. m_shapes.emplace_back( new SHAPE_SEGMENT( center - arm, center + arm ) );
  1041. }
  1042. static struct DIMENSION_DESC
  1043. {
  1044. DIMENSION_DESC()
  1045. {
  1046. ENUM_MAP<DIM_PRECISION>::Instance()
  1047. .Map( DIM_PRECISION::X, _HKI( "0" ) )
  1048. .Map( DIM_PRECISION::X_X, _HKI( "0.0" ) )
  1049. .Map( DIM_PRECISION::X_XX, _HKI( "0.00" ) )
  1050. .Map( DIM_PRECISION::X_XXX, _HKI( "0.000" ) )
  1051. .Map( DIM_PRECISION::X_XXXX, _HKI( "0.0000" ) )
  1052. .Map( DIM_PRECISION::X_XXXXX, _HKI( "0.00000" ) )
  1053. .Map( DIM_PRECISION::V_VV, _HKI( "0.00 in / 0 mils / 0.0 mm" ) )
  1054. .Map( DIM_PRECISION::V_VVV, _HKI( "0.000 / 0 / 0.00" ) )
  1055. .Map( DIM_PRECISION::V_VVVV, _HKI( "0.0000 / 0.0 / 0.000" ) )
  1056. .Map( DIM_PRECISION::V_VVVVV, _HKI( "0.00000 / 0.00 / 0.0000" ) );
  1057. ENUM_MAP<DIM_UNITS_FORMAT>::Instance()
  1058. .Map( DIM_UNITS_FORMAT::NO_SUFFIX, _HKI( "1234.0" ) )
  1059. .Map( DIM_UNITS_FORMAT::BARE_SUFFIX, _HKI( "1234.0 mm" ) )
  1060. .Map( DIM_UNITS_FORMAT::PAREN_SUFFIX, _HKI( "1234.0 (mm)" ) );
  1061. ENUM_MAP<DIM_UNITS_MODE>::Instance()
  1062. .Map( DIM_UNITS_MODE::INCHES, _HKI( "Inches" ) )
  1063. .Map( DIM_UNITS_MODE::MILS, _HKI( "Mils" ) )
  1064. .Map( DIM_UNITS_MODE::MILLIMETRES, _HKI( "Millimeters" ) )
  1065. .Map( DIM_UNITS_MODE::AUTOMATIC, _HKI( "Automatic" ) );
  1066. PROPERTY_MANAGER& propMgr = PROPERTY_MANAGER::Instance();
  1067. REGISTER_TYPE( PCB_DIMENSION_BASE );
  1068. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIMENSION_BASE, PCB_TEXT> );
  1069. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIMENSION_BASE, BOARD_ITEM> );
  1070. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIMENSION_BASE, EDA_TEXT> );
  1071. propMgr.InheritsAfter( TYPE_HASH( PCB_DIMENSION_BASE ), TYPE_HASH( PCB_TEXT ) );
  1072. propMgr.InheritsAfter( TYPE_HASH( PCB_DIMENSION_BASE ), TYPE_HASH( BOARD_ITEM ) );
  1073. propMgr.InheritsAfter( TYPE_HASH( PCB_DIMENSION_BASE ), TYPE_HASH( EDA_TEXT ) );
  1074. propMgr.Mask( TYPE_HASH( PCB_DIMENSION_BASE ), TYPE_HASH( EDA_TEXT ), _HKI( "Orientation" ) );
  1075. const wxString groupDimension = _HKI( "Dimension Properties" );
  1076. auto isLeader =
  1077. []( INSPECTABLE* aItem ) -> bool
  1078. {
  1079. return dynamic_cast<PCB_DIM_LEADER*>( aItem ) != nullptr;
  1080. };
  1081. auto isNotLeader =
  1082. []( INSPECTABLE* aItem ) -> bool
  1083. {
  1084. return dynamic_cast<PCB_DIM_LEADER*>( aItem ) == nullptr;
  1085. };
  1086. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, wxString>( _HKI( "Prefix" ),
  1087. &PCB_DIMENSION_BASE::ChangePrefix, &PCB_DIMENSION_BASE::GetPrefix ),
  1088. groupDimension )
  1089. .SetAvailableFunc( isNotLeader );
  1090. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, wxString>( _HKI( "Suffix" ),
  1091. &PCB_DIMENSION_BASE::ChangeSuffix, &PCB_DIMENSION_BASE::GetSuffix ),
  1092. groupDimension )
  1093. .SetAvailableFunc( isNotLeader );
  1094. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, wxString>( _HKI( "Override Text" ),
  1095. &PCB_DIMENSION_BASE::ChangeOverrideText, &PCB_DIMENSION_BASE::GetOverrideText ),
  1096. groupDimension )
  1097. .SetAvailableFunc( isNotLeader );
  1098. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, wxString>( _HKI( "Text" ),
  1099. &PCB_DIMENSION_BASE::ChangeOverrideText, &PCB_DIMENSION_BASE::GetOverrideText ),
  1100. groupDimension )
  1101. .SetAvailableFunc( isLeader );
  1102. propMgr.AddProperty( new PROPERTY_ENUM<PCB_DIMENSION_BASE, DIM_UNITS_MODE>( _HKI( "Units" ),
  1103. &PCB_DIMENSION_BASE::ChangeUnitsMode, &PCB_DIMENSION_BASE::GetUnitsMode ),
  1104. groupDimension )
  1105. .SetAvailableFunc( isNotLeader );
  1106. propMgr.AddProperty( new PROPERTY_ENUM<PCB_DIMENSION_BASE, DIM_UNITS_FORMAT>( _HKI( "Units Format" ),
  1107. &PCB_DIMENSION_BASE::ChangeUnitsFormat, &PCB_DIMENSION_BASE::GetUnitsFormat ),
  1108. groupDimension )
  1109. .SetAvailableFunc( isNotLeader );
  1110. propMgr.AddProperty( new PROPERTY_ENUM<PCB_DIMENSION_BASE, DIM_PRECISION>( _HKI( "Precision" ),
  1111. &PCB_DIMENSION_BASE::ChangePrecision, &PCB_DIMENSION_BASE::GetPrecision ),
  1112. groupDimension )
  1113. .SetAvailableFunc( isNotLeader );
  1114. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, bool>( _HKI( "Suppress Trailing Zeroes" ),
  1115. &PCB_DIMENSION_BASE::ChangeSuppressZeroes, &PCB_DIMENSION_BASE::GetSuppressZeroes ),
  1116. groupDimension )
  1117. .SetAvailableFunc( isNotLeader );
  1118. const wxString groupText = _HKI( "Text Properties" );
  1119. const auto isTextOrientationWriteable =
  1120. []( INSPECTABLE* aItem ) -> bool
  1121. {
  1122. return !static_cast<PCB_DIMENSION_BASE*>( aItem )->GetKeepTextAligned();
  1123. };
  1124. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, bool>( _HKI( "Keep Aligned with Dimension" ),
  1125. &PCB_DIMENSION_BASE::ChangeKeepTextAligned,
  1126. &PCB_DIMENSION_BASE::GetKeepTextAligned ),
  1127. groupText );
  1128. propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, double>( _HKI( "Orientation" ),
  1129. &PCB_DIMENSION_BASE::ChangeTextAngleDegrees,
  1130. &PCB_DIMENSION_BASE::GetTextAngleDegreesProp,
  1131. PROPERTY_DISPLAY::PT_DEGREE ),
  1132. groupText )
  1133. .SetWriteableFunc( isTextOrientationWriteable );
  1134. }
  1135. } _DIMENSION_DESC;
  1136. ENUM_TO_WXANY( DIM_PRECISION )
  1137. ENUM_TO_WXANY( DIM_UNITS_FORMAT )
  1138. ENUM_TO_WXANY( DIM_UNITS_MODE )
  1139. static struct ALIGNED_DIMENSION_DESC
  1140. {
  1141. ALIGNED_DIMENSION_DESC()
  1142. {
  1143. PROPERTY_MANAGER& propMgr = PROPERTY_MANAGER::Instance();
  1144. REGISTER_TYPE( PCB_DIM_ALIGNED );
  1145. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ALIGNED, BOARD_ITEM> );
  1146. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ALIGNED, EDA_TEXT> );
  1147. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ALIGNED, PCB_TEXT> );
  1148. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ALIGNED, PCB_DIMENSION_BASE> );
  1149. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( BOARD_ITEM ) );
  1150. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( EDA_TEXT ) );
  1151. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( PCB_TEXT ) );
  1152. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( PCB_DIMENSION_BASE ) );
  1153. const wxString groupDimension = _HKI( "Dimension Properties" );
  1154. propMgr.AddProperty( new PROPERTY<PCB_DIM_ALIGNED, int>( _HKI( "Crossbar Height" ),
  1155. &PCB_DIM_ALIGNED::ChangeHeight, &PCB_DIM_ALIGNED::GetHeight,
  1156. PROPERTY_DISPLAY::PT_SIZE ),
  1157. groupDimension );
  1158. propMgr.AddProperty( new PROPERTY<PCB_DIM_ALIGNED, int>( _HKI( "Extension Line Overshoot" ),
  1159. &PCB_DIM_ALIGNED::ChangeExtensionHeight, &PCB_DIM_ALIGNED::GetExtensionHeight,
  1160. PROPERTY_DISPLAY::PT_SIZE ),
  1161. groupDimension );
  1162. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( EDA_TEXT ),
  1163. _HKI( "Visible" ),
  1164. []( INSPECTABLE* aItem ) { return false; } );
  1165. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( EDA_TEXT ),
  1166. _HKI( "Text" ),
  1167. []( INSPECTABLE* aItem ) { return false; } );
  1168. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( EDA_TEXT ),
  1169. _HKI( "Vertical Justification" ),
  1170. []( INSPECTABLE* aItem ) { return false; } );
  1171. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( EDA_TEXT ),
  1172. _HKI( "Hyperlink" ),
  1173. []( INSPECTABLE* aItem ) { return false; } );
  1174. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ALIGNED ), TYPE_HASH( BOARD_ITEM ),
  1175. _HKI( "Knockout" ),
  1176. []( INSPECTABLE* aItem ) { return false; } );
  1177. }
  1178. } ALIGNED_DIMENSION_DESC;
  1179. static struct ORTHOGONAL_DIMENSION_DESC
  1180. {
  1181. ORTHOGONAL_DIMENSION_DESC()
  1182. {
  1183. PROPERTY_MANAGER& propMgr = PROPERTY_MANAGER::Instance();
  1184. REGISTER_TYPE( PCB_DIM_ORTHOGONAL );
  1185. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ORTHOGONAL, BOARD_ITEM> );
  1186. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ORTHOGONAL, EDA_TEXT> );
  1187. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ORTHOGONAL, PCB_TEXT> );
  1188. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ORTHOGONAL, PCB_DIMENSION_BASE> );
  1189. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_ORTHOGONAL, PCB_DIM_ALIGNED> );
  1190. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( BOARD_ITEM ) );
  1191. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( EDA_TEXT ) );
  1192. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( PCB_TEXT ) );
  1193. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( PCB_DIMENSION_BASE ) );
  1194. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( PCB_DIM_ALIGNED ) );
  1195. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( EDA_TEXT ),
  1196. _HKI( "Visible" ),
  1197. []( INSPECTABLE* aItem ) { return false; } );
  1198. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( EDA_TEXT ),
  1199. _HKI( "Text" ),
  1200. []( INSPECTABLE* aItem ) { return false; } );
  1201. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( EDA_TEXT ),
  1202. _HKI( "Vertical Justification" ),
  1203. []( INSPECTABLE* aItem ) { return false; } );
  1204. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( EDA_TEXT ),
  1205. _HKI( "Hyperlink" ),
  1206. []( INSPECTABLE* aItem ) { return false; } );
  1207. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_ORTHOGONAL ), TYPE_HASH( BOARD_ITEM ),
  1208. _HKI( "Knockout" ),
  1209. []( INSPECTABLE* aItem ) { return false; } );
  1210. }
  1211. } ORTHOGONAL_DIMENSION_DESC;
  1212. static struct RADIAL_DIMENSION_DESC
  1213. {
  1214. RADIAL_DIMENSION_DESC()
  1215. {
  1216. PROPERTY_MANAGER& propMgr = PROPERTY_MANAGER::Instance();
  1217. REGISTER_TYPE( PCB_DIM_RADIAL );
  1218. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_RADIAL, BOARD_ITEM> );
  1219. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_RADIAL, EDA_TEXT> );
  1220. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_RADIAL, PCB_TEXT> );
  1221. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_RADIAL, PCB_DIMENSION_BASE> );
  1222. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( BOARD_ITEM ) );
  1223. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( EDA_TEXT ) );
  1224. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( PCB_TEXT ) );
  1225. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( PCB_DIMENSION_BASE ) );
  1226. const wxString groupDimension = _HKI( "Dimension Properties" );
  1227. propMgr.AddProperty( new PROPERTY<PCB_DIM_RADIAL, int>( _HKI( "Leader Length" ),
  1228. &PCB_DIM_RADIAL::ChangeLeaderLength, &PCB_DIM_RADIAL::GetLeaderLength,
  1229. PROPERTY_DISPLAY::PT_SIZE ),
  1230. groupDimension );
  1231. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( EDA_TEXT ),
  1232. _HKI( "Visible" ),
  1233. []( INSPECTABLE* aItem ) { return false; } );
  1234. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( EDA_TEXT ),
  1235. _HKI( "Text" ),
  1236. []( INSPECTABLE* aItem ) { return false; } );
  1237. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( EDA_TEXT ),
  1238. _HKI( "Vertical Justification" ),
  1239. []( INSPECTABLE* aItem ) { return false; } );
  1240. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( EDA_TEXT ),
  1241. _HKI( "Hyperlink" ),
  1242. []( INSPECTABLE* aItem ) { return false; } );
  1243. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_RADIAL ), TYPE_HASH( BOARD_ITEM ),
  1244. _HKI( "Knockout" ),
  1245. []( INSPECTABLE* aItem ) { return false; } );
  1246. }
  1247. } RADIAL_DIMENSION_DESC;
  1248. static struct LEADER_DIMENSION_DESC
  1249. {
  1250. LEADER_DIMENSION_DESC()
  1251. {
  1252. ENUM_MAP<DIM_TEXT_BORDER>::Instance()
  1253. .Map( DIM_TEXT_BORDER::NONE, _HKI( "None" ) )
  1254. .Map( DIM_TEXT_BORDER::RECTANGLE, _HKI( "Rectangle" ) )
  1255. .Map( DIM_TEXT_BORDER::CIRCLE, _HKI( "Circle" ) );
  1256. PROPERTY_MANAGER& propMgr = PROPERTY_MANAGER::Instance();
  1257. REGISTER_TYPE( PCB_DIM_LEADER );
  1258. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_LEADER, BOARD_ITEM> );
  1259. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_LEADER, EDA_TEXT> );
  1260. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_LEADER, PCB_TEXT> );
  1261. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_LEADER, PCB_DIMENSION_BASE> );
  1262. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( BOARD_ITEM ) );
  1263. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( EDA_TEXT ) );
  1264. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( PCB_TEXT ) );
  1265. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( PCB_DIMENSION_BASE ) );
  1266. const wxString groupDimension = _HKI( "Dimension Properties" );
  1267. propMgr.AddProperty( new PROPERTY_ENUM<PCB_DIM_LEADER, DIM_TEXT_BORDER>( _HKI( "Text Frame" ),
  1268. &PCB_DIM_LEADER::ChangeTextBorder, &PCB_DIM_LEADER::GetTextBorder ),
  1269. groupDimension );
  1270. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( EDA_TEXT ),
  1271. _HKI( "Visible" ),
  1272. []( INSPECTABLE* aItem ) { return false; } );
  1273. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( EDA_TEXT ),
  1274. _HKI( "Text" ),
  1275. []( INSPECTABLE* aItem ) { return false; } );
  1276. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( EDA_TEXT ),
  1277. _HKI( "Vertical Justification" ),
  1278. []( INSPECTABLE* aItem ) { return false; } );
  1279. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( EDA_TEXT ),
  1280. _HKI( "Hyperlink" ),
  1281. []( INSPECTABLE* aItem ) { return false; } );
  1282. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_LEADER ), TYPE_HASH( BOARD_ITEM ),
  1283. _HKI( "Knockout" ),
  1284. []( INSPECTABLE* aItem ) { return false; } );
  1285. }
  1286. } LEADER_DIMENSION_DESC;
  1287. ENUM_TO_WXANY( DIM_TEXT_BORDER )
  1288. static struct CENTER_DIMENSION_DESC
  1289. {
  1290. CENTER_DIMENSION_DESC()
  1291. {
  1292. PROPERTY_MANAGER& propMgr = PROPERTY_MANAGER::Instance();
  1293. REGISTER_TYPE( PCB_DIM_CENTER );
  1294. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_CENTER, BOARD_ITEM> );
  1295. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_CENTER, EDA_TEXT> );
  1296. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_CENTER, PCB_TEXT> );
  1297. propMgr.AddTypeCast( new TYPE_CAST<PCB_DIM_CENTER, PCB_DIMENSION_BASE> );
  1298. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( BOARD_ITEM ) );
  1299. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( EDA_TEXT ) );
  1300. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( PCB_TEXT ) );
  1301. propMgr.InheritsAfter( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( PCB_DIMENSION_BASE ) );
  1302. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( EDA_TEXT ),
  1303. _HKI( "Visible" ),
  1304. []( INSPECTABLE* aItem ) { return false; } );
  1305. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( EDA_TEXT ),
  1306. _HKI( "Text" ),
  1307. []( INSPECTABLE* aItem ) { return false; } );
  1308. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( EDA_TEXT ),
  1309. _HKI( "Vertical Justification" ),
  1310. []( INSPECTABLE* aItem ) { return false; } );
  1311. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( EDA_TEXT ),
  1312. _HKI( "Hyperlink" ),
  1313. []( INSPECTABLE* aItem ) { return false; } );
  1314. propMgr.OverrideAvailability( TYPE_HASH( PCB_DIM_CENTER ), TYPE_HASH( BOARD_ITEM ),
  1315. _HKI( "Knockout" ),
  1316. []( INSPECTABLE* aItem ) { return false; } );
  1317. }
  1318. } CENTER_DIMENSION_DESC;