/*******************************************************************/ /* dialog_pad_properties.cpp: Pad editing functions and dialog box */ /* see also dialog_pad_properties.xxx (built with wxFormBuilder) */ /*******************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include // list of pad shapes. static PAD_SHAPE_T CodeShape[] = { PAD_CIRCLE, PAD_OVAL, PAD_RECT, PAD_TRAPEZOID }; #define NBTYPES 4 static PAD_ATTR_T CodeType[NBTYPES] = { PAD_STANDARD, PAD_SMD, PAD_CONN, PAD_HOLE_NOT_PLATED }; // Default mask layers setup for pads according to the pad type static long Std_Pad_Layers[NBTYPES] = { // PAD_STANDARD: PAD_STANDARD_DEFAULT_LAYERS, // PAD_CONN: PAD_CONN_DEFAULT_LAYERS, // PAD_SMD: PAD_SMD_DEFAULT_LAYERS, //PAD_HOLE_NOT_PLATED: PAD_HOLE_NOT_PLATED_DEFAULT_LAYERS }; /** * class DIALOG_PAD_PROPERTIES, derived from DIALOG_PAD_PROPERTIES_BASE, * created by wxFormBuilder */ class DIALOG_PAD_PROPERTIES : public DIALOG_PAD_PROPERTIES_BASE { public: DIALOG_PAD_PROPERTIES( PCB_BASE_FRAME* aParent, D_PAD* aPad ); ~DIALOG_PAD_PROPERTIES() { delete m_dummyPad; } private: PCB_BASE_FRAME* m_Parent; D_PAD* m_CurrentPad; // pad currently being edited D_PAD* m_dummyPad; // a working copy used to show changes BOARD* m_Board; D_PAD& m_Pad_Master; bool m_isFlipped; // true if the parent footprint (therefore pads) is flipped (mirrored) // in this case, some Y coordinates values must be negated bool m_canUpdate; void initValues(); void OnPadShapeSelection( wxCommandEvent& event ); void OnDrillShapeSelected( wxCommandEvent& event ); void PadOrientEvent( wxCommandEvent& event ); void PadTypeSelected( wxCommandEvent& event ); void PadPropertiesAccept( wxCommandEvent& event ); void SetPadLayersList( long layer_mask ); void OnSetLayers( wxCommandEvent& event ); void OnCancelButtonClick( wxCommandEvent& event ); void OnPaintShowPanel( wxPaintEvent& event ); bool TransfertDataToPad( D_PAD* aPad, bool aPromptOnError = false ); void OnValuesChanged( wxCommandEvent& event ); }; DIALOG_PAD_PROPERTIES::DIALOG_PAD_PROPERTIES( PCB_BASE_FRAME* aParent, D_PAD* aPad ) : DIALOG_PAD_PROPERTIES_BASE( aParent ), m_Pad_Master( aParent->GetBoard()->GetDesignSettings().m_Pad_Master ) { m_canUpdate = false; m_Parent = aParent; m_CurrentPad = aPad; m_Board = m_Parent->GetBoard(); m_dummyPad = new D_PAD( (MODULE*) NULL ); if( aPad ) m_dummyPad->Copy( aPad ); else m_dummyPad->Copy( &m_Pad_Master ); initValues(); m_sdbSizer1OK->SetDefault(); GetSizer()->SetSizeHints( this ); Center(); m_canUpdate = true; } void DIALOG_PAD_PROPERTIES::OnPaintShowPanel( wxPaintEvent& event ) { wxPaintDC dc( m_panelShowPad ); PAD_DRAWINFO drawInfo; int color = 0; if( m_dummyPad->GetLayerMask() & LAYER_FRONT ) { color = m_Board->GetVisibleElementColor( PAD_FR_VISIBLE ); } if( m_dummyPad->GetLayerMask() & LAYER_BACK ) { color |= m_Board->GetVisibleElementColor( PAD_BK_VISIBLE ); } if( color == 0 ) color = LIGHTGRAY; drawInfo.m_Color = color; drawInfo.m_HoleColor = DARKGRAY; drawInfo.m_Offset = m_dummyPad->GetPosition(); drawInfo.m_Display_padnum = true; drawInfo.m_Display_netname = true; if( m_dummyPad->GetAttribute() == PAD_HOLE_NOT_PLATED ) drawInfo.m_ShowNotPlatedHole = true; // Shows the local pad clearance drawInfo.m_PadClearance = m_dummyPad->GetLocalClearance(); wxSize dc_size = dc.GetSize(); dc.SetDeviceOrigin( dc_size.x / 2, dc_size.y / 2 ); // Calculate a suitable scale to fit the available draw area int dim = m_dummyPad->GetSize().x + ABS( m_dummyPad->GetDelta().y); if( m_dummyPad->GetLocalClearance() > 0 ) dim += m_dummyPad->GetLocalClearance() * 2; double scale = (double) dc_size.x / dim; dim = m_dummyPad->GetSize().y + ABS( m_dummyPad->GetDelta().x); if( m_dummyPad->GetLocalClearance() > 0 ) dim += m_dummyPad->GetLocalClearance() * 2; double altscale = (double) dc_size.y / dim; scale = MIN( scale, altscale ); // Give a margin scale *= 0.7; dc.SetUserScale( scale, scale ); GRResetPenAndBrush( &dc ); m_dummyPad->DrawShape( NULL, &dc, drawInfo ); event.Skip(); } void PCB_BASE_FRAME::InstallPadOptionsFrame( D_PAD* Pad ) { DIALOG_PAD_PROPERTIES dlg( this, Pad ); dlg.ShowModal(); } void DIALOG_PAD_PROPERTIES::initValues() { SetFocus(); // Required under wxGTK if we want to dismiss the dialog with the ESC key int internalUnits = m_Parent->GetInternalUnits(); wxString msg; double angle; m_isFlipped = false; if( m_CurrentPad ) { MODULE* module = m_CurrentPad->GetParent(); if( module->GetLayer() == LAYER_N_BACK ) { m_isFlipped = true; m_staticModuleSideValue->SetLabel( _( "Back side (footprint is mirrored)" ) ); } msg.Printf( wxT( "%.1f" ), (double) module->GetOrientation() / 10 ); m_staticModuleRotValue->SetLabel( msg ); } if( m_isFlipped ) { wxPoint pt = m_dummyPad->GetOffset(); NEGATE( pt.y ); m_dummyPad->SetOffset( pt ); wxSize sz = m_dummyPad->GetDelta(); NEGATE( sz.y ); m_dummyPad->SetDelta( sz ); // flip pad's layers m_dummyPad->SetLayerMask( ChangeSideMaskLayer( m_dummyPad->GetLayerMask() ) ); } m_staticTextWarningPadFlipped->Show(m_isFlipped); m_PadNumCtrl->SetValue( m_dummyPad->GetPadName() ); m_PadNetNameCtrl->SetValue( m_dummyPad->GetNetname() ); // Display current unit name in dialog: m_PadPosX_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadPosY_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadDrill_X_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadDrill_Y_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadShapeSizeX_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadShapeSizeY_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadShapeOffsetX_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadShapeOffsetY_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadShapeDelta_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_PadLengthDie_Unit->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_NetClearanceUnits->SetLabel( GetUnitsLabel( g_UserUnit ) ); // Display current pad masks clearances units m_NetClearanceUnits->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_SolderMaskMarginUnits->SetLabel( GetUnitsLabel( g_UserUnit ) ); m_SolderPasteMarginUnits->SetLabel( GetUnitsLabel( g_UserUnit ) ); // Display current pad parameters units: PutValueInLocalUnits( *m_PadPosition_X_Ctrl, m_dummyPad->GetPosition().x, internalUnits ); PutValueInLocalUnits( *m_PadPosition_Y_Ctrl, m_dummyPad->GetPosition().y, internalUnits ); PutValueInLocalUnits( *m_PadDrill_X_Ctrl, m_dummyPad->GetDrillSize().x, internalUnits ); PutValueInLocalUnits( *m_PadDrill_Y_Ctrl, m_dummyPad->GetDrillSize().y, internalUnits ); PutValueInLocalUnits( *m_ShapeSize_X_Ctrl, m_dummyPad->GetSize().x, internalUnits ); PutValueInLocalUnits( *m_ShapeSize_Y_Ctrl, m_dummyPad->GetSize().y, internalUnits ); PutValueInLocalUnits( *m_ShapeOffset_X_Ctrl, m_dummyPad->GetOffset().x, internalUnits ); PutValueInLocalUnits( *m_ShapeOffset_Y_Ctrl, m_dummyPad->GetOffset().y, internalUnits ); if( m_dummyPad->GetDelta().x ) { PutValueInLocalUnits( *m_ShapeDelta_Ctrl, m_dummyPad->GetDelta().x, internalUnits ); m_radioBtnDeltaXdir->SetValue(true); } else { PutValueInLocalUnits( *m_ShapeDelta_Ctrl, m_dummyPad->GetDelta().y, internalUnits ); m_radioBtnDeltaYdir->SetValue(true); } PutValueInLocalUnits( *m_LengthDieCtrl, m_dummyPad->GetDieLength(), internalUnits ); PutValueInLocalUnits( *m_NetClearanceValueCtrl, m_dummyPad->GetLocalClearance(), internalUnits ); PutValueInLocalUnits( *m_SolderMaskMarginCtrl, m_dummyPad->GetLocalSolderMaskMargin(), internalUnits ); // These 2 parameters are usually < 0, so prepare entering a negative value, if current is 0 PutValueInLocalUnits( *m_SolderPasteMarginCtrl, m_dummyPad->GetLocalSolderPasteMargin(), internalUnits ); if( m_dummyPad->GetLocalSolderPasteMargin() == 0 ) m_SolderPasteMarginCtrl->SetValue( wxT( "-" ) + m_SolderPasteMarginCtrl->GetValue() ); msg.Printf( wxT( "%.1f" ), m_dummyPad->GetLocalSolderPasteMarginRatio() * 100.0 ); if( m_dummyPad->GetLocalSolderPasteMarginRatio() == 0.0 && msg[0] == '0' ) // Sometimes Printf adds a sign if the value is small m_SolderPasteMarginRatioCtrl->SetValue( wxT( "-" ) + msg ); else m_SolderPasteMarginRatioCtrl->SetValue( msg ); switch( m_dummyPad->GetZoneConnection() ) { default: case UNDEFINED_CONNECTION: m_ZoneConnectionChoice->SetSelection( 0 ); break; case PAD_IN_ZONE: m_ZoneConnectionChoice->SetSelection( 1 ); break; case THERMAL_PAD: m_ZoneConnectionChoice->SetSelection( 2 ); break; case PAD_NOT_IN_ZONE: m_ZoneConnectionChoice->SetSelection( 3 ); break; } if( m_CurrentPad ) { MODULE* module = m_CurrentPad->GetParent(); angle = m_CurrentPad->GetOrientation() - module->GetOrientation(); if( m_isFlipped ) NEGATE( angle ); m_dummyPad->SetOrientation( angle ); } angle = m_dummyPad->GetOrientation(); NORMALIZE_ANGLE_180( angle ); // ? normalizing is in D_PAD::SetOrientation() // Set layers used by this pad: : SetPadLayersList( m_dummyPad->GetLayerMask() ); // Pad Orient switch( int( angle ) ) { case 0: m_PadOrient->SetSelection( 0 ); break; case 900: m_PadOrient->SetSelection( 1 ); break; case - 900: m_PadOrient->SetSelection( 2 ); break; case 1800: case -1800: m_PadOrient->SetSelection( 3 ); break; default: m_PadOrient->SetSelection( 4 ); break; } switch( m_dummyPad->GetShape() ) { default: case PAD_CIRCLE: m_PadShape->SetSelection( 0 ); break; case PAD_OVAL: m_PadShape->SetSelection( 1 ); break; case PAD_RECT: m_PadShape->SetSelection( 2 ); break; case PAD_TRAPEZOID: m_PadShape->SetSelection( 3 ); break; } msg.Printf( wxT( "%g" ), angle ); m_PadOrientCtrl->SetValue( msg ); // Type of pad selection m_PadType->SetSelection( 0 ); for( int ii = 0; ii < NBTYPES; ii++ ) { if( CodeType[ii] == m_dummyPad->GetAttribute() ) { m_PadType->SetSelection( ii ); break; } } // Enable/disable Pad name,and pad length die // (disable for NPTH pads (mechanical pads) bool enable = m_dummyPad->GetAttribute() != PAD_HOLE_NOT_PLATED; m_PadNumCtrl->Enable( enable ); m_PadNetNameCtrl->Enable( enable ); m_LengthDieCtrl->Enable( enable ); if( m_dummyPad->GetDrillShape() != PAD_OVAL ) m_DrillShapeCtrl->SetSelection( 0 ); else m_DrillShapeCtrl->SetSelection( 1 ); // Setup layers names from board m_rbCopperLayersSel->SetString( 0, m_Board->GetLayerName( LAYER_N_FRONT ) ); m_rbCopperLayersSel->SetString( 1, m_Board->GetLayerName( LAYER_N_BACK ) ); m_PadLayerAdhCmp->SetLabel( m_Board->GetLayerName( ADHESIVE_N_FRONT ) ); m_PadLayerAdhCu->SetLabel( m_Board->GetLayerName( ADHESIVE_N_BACK ) ); m_PadLayerPateCmp->SetLabel( m_Board->GetLayerName( SOLDERPASTE_N_FRONT ) ); m_PadLayerPateCu->SetLabel( m_Board->GetLayerName( SOLDERPASTE_N_BACK ) ); m_PadLayerSilkCmp->SetLabel( m_Board->GetLayerName( SILKSCREEN_N_FRONT ) ); m_PadLayerSilkCu->SetLabel( m_Board->GetLayerName( SILKSCREEN_N_BACK ) ); m_PadLayerMaskCmp->SetLabel( m_Board->GetLayerName( SOLDERMASK_N_FRONT ) ); m_PadLayerMaskCu->SetLabel( m_Board->GetLayerName( SOLDERMASK_N_BACK ) ); m_PadLayerECO1->SetLabel( m_Board->GetLayerName( ECO1_N ) ); m_PadLayerECO2->SetLabel( m_Board->GetLayerName( ECO2_N ) ); m_PadLayerDraft->SetLabel( m_Board->GetLayerName( DRAW_N ) ); // Update some dialog widgets state (Enable/disable options): wxCommandEvent cmd_event; OnPadShapeSelection( cmd_event ); OnDrillShapeSelected( cmd_event ); } void DIALOG_PAD_PROPERTIES::OnPadShapeSelection( wxCommandEvent& event ) { switch( m_PadShape->GetSelection() ) { case 0: //CIRCLE: m_ShapeDelta_Ctrl->Enable( false ); m_radioBtnDeltaXdir->Enable( false ); m_radioBtnDeltaYdir->Enable( false ); m_ShapeSize_Y_Ctrl->Enable( false ); break; case 1: //OVALE: m_ShapeDelta_Ctrl->Enable( false ); m_radioBtnDeltaXdir->Enable( false ); m_radioBtnDeltaYdir->Enable( false ); m_ShapeSize_Y_Ctrl->Enable( true ); break; case 2: // PAD_RECT: m_ShapeDelta_Ctrl->Enable( false ); m_radioBtnDeltaXdir->Enable( false ); m_radioBtnDeltaYdir->Enable( false ); m_ShapeSize_Y_Ctrl->Enable( true ); break; case 3: //TRAPEZE: m_ShapeDelta_Ctrl->Enable( true ); m_radioBtnDeltaXdir->Enable( true ); m_radioBtnDeltaYdir->Enable( true ); m_ShapeSize_Y_Ctrl->Enable( true ); break; } TransfertDataToPad( m_dummyPad ); m_panelShowPad->Refresh(); } void DIALOG_PAD_PROPERTIES::OnDrillShapeSelected( wxCommandEvent& event ) { if( m_PadType->GetSelection() == 1 || m_PadType->GetSelection() == 2 ) { // pad type = SMD or CONN: no hole allowed m_PadDrill_X_Ctrl->Enable( false ); m_PadDrill_Y_Ctrl->Enable( false ); } else { switch( m_DrillShapeCtrl->GetSelection() ) { case 0: //CIRCLE: m_PadDrill_X_Ctrl->Enable( true ); m_PadDrill_Y_Ctrl->Enable( false ); break; case 1: //OVALE: m_PadDrill_X_Ctrl->Enable( true ); m_PadDrill_Y_Ctrl->Enable( true ); break; } } TransfertDataToPad( m_dummyPad ); m_panelShowPad->Refresh(); } void DIALOG_PAD_PROPERTIES::PadOrientEvent( wxCommandEvent& event ) { switch( m_PadOrient->GetSelection() ) { case 0: m_dummyPad->SetOrientation( 0 ); break; case 1: m_dummyPad->SetOrientation( 900 ); break; case 2: m_dummyPad->SetOrientation( -900 ); break; case 3: m_dummyPad->SetOrientation( 1800 ); break; default: break; } wxString msg; msg.Printf( wxT( "%g" ), m_dummyPad->GetOrientation() ); m_PadOrientCtrl->SetValue( msg ); TransfertDataToPad( m_dummyPad ); m_panelShowPad->Refresh(); } void DIALOG_PAD_PROPERTIES::PadTypeSelected( wxCommandEvent& event ) /* Adjust the better mask layer according to the selected pad type */ { long layer_mask; int ii; ii = m_PadType->GetSelection(); if( (ii < 0) || ( ii >= NBTYPES) ) ii = 0; layer_mask = Std_Pad_Layers[ii]; SetPadLayersList( layer_mask ); // Enable/disable drill dialog items: event.SetId( m_DrillShapeCtrl->GetSelection() ); OnDrillShapeSelected( event ); // Enable/disable Pad name,and pad length die // (disable for NPTH pads (mechanical pads) bool enable = ii != 3; m_PadNumCtrl->Enable( enable ); m_PadNetNameCtrl->Enable( enable ); m_LengthDieCtrl->Enable( enable ); } void DIALOG_PAD_PROPERTIES::SetPadLayersList( long layer_mask ) /** SetPadLayersList * Update the CheckBoxes state in pad layers list, * @param layer_mask = pad layer mask (ORed layers bit mask) */ { if( ( layer_mask & ALL_CU_LAYERS ) == LAYER_FRONT ) m_rbCopperLayersSel->SetSelection(0); else if( ( layer_mask & ALL_CU_LAYERS ) == LAYER_BACK) m_rbCopperLayersSel->SetSelection(1); else if( ( layer_mask & ALL_CU_LAYERS ) != 0 ) m_rbCopperLayersSel->SetSelection(2); else m_rbCopperLayersSel->SetSelection(3); m_PadLayerAdhCmp->SetValue( ( layer_mask & ADHESIVE_LAYER_FRONT ) ); m_PadLayerAdhCu->SetValue( ( layer_mask & ADHESIVE_LAYER_BACK ) ); m_PadLayerPateCmp->SetValue( ( layer_mask & SOLDERPASTE_LAYER_FRONT ) ); m_PadLayerPateCu->SetValue( ( layer_mask & SOLDERPASTE_LAYER_BACK ) ); m_PadLayerSilkCmp->SetValue( ( layer_mask & SILKSCREEN_LAYER_FRONT ) ); m_PadLayerSilkCu->SetValue( ( layer_mask & SILKSCREEN_LAYER_BACK ) ); m_PadLayerMaskCmp->SetValue( ( layer_mask & SOLDERMASK_LAYER_FRONT ) ); m_PadLayerMaskCu->SetValue( ( layer_mask & SOLDERMASK_LAYER_BACK ) ); m_PadLayerECO1->SetValue( ( layer_mask & ECO1_LAYER ) ); m_PadLayerECO2->SetValue( ( layer_mask & ECO2_LAYER ) ); m_PadLayerDraft->SetValue( ( layer_mask & DRAW_LAYER ) ); } // Called when select/deselect a layer. void DIALOG_PAD_PROPERTIES::OnSetLayers( wxCommandEvent& event ) { TransfertDataToPad( m_dummyPad ); m_panelShowPad->Refresh(); } void DIALOG_PAD_PROPERTIES::PadPropertiesAccept( wxCommandEvent& event ) /* Updates the different parameters for the component being edited. */ { bool rastnestIsChanged = false; int isign = m_isFlipped ? -1 : 1; bool success = TransfertDataToPad( m_dummyPad, true ); if( !success ) // An error on parameters has occured return; TransfertDataToPad( &m_Pad_Master, false ); if( m_CurrentPad ) // Set current Pad parameters { wxSize size; MODULE* module = m_CurrentPad->GetParent(); m_Parent->SaveCopyInUndoList( module, UR_CHANGED ); module->m_LastEdit_Time = time( NULL ); // redraw the area where the pad was, without pad (delete pad on screen) m_CurrentPad->SetFlags( DO_NOT_DRAW ); m_Parent->GetCanvas()->RefreshDrawingRect( m_CurrentPad->GetBoundingBox() ); m_CurrentPad->ClearFlags( DO_NOT_DRAW ); // Update values m_CurrentPad->SetShape( m_Pad_Master.GetShape() ); m_CurrentPad->SetAttribute( m_Pad_Master.GetAttribute() ); if( m_CurrentPad->GetPosition() != m_Pad_Master.GetPosition() ) { m_CurrentPad->SetPosition( m_Pad_Master.GetPosition() ); rastnestIsChanged = true; } // compute the pos 0 value, i.e. pad position for module with orientation = 0 // i.e. relative to module origin (module position) wxPoint pt = m_CurrentPad->GetPosition() - module->GetPosition(); RotatePoint( &pt, -module->GetOrientation() ); m_CurrentPad->SetPos0( pt ); m_CurrentPad->SetOrientation( m_Pad_Master.GetOrientation() * isign + module->GetOrientation() ); m_CurrentPad->SetSize( m_Pad_Master.GetSize() ); size = m_Pad_Master.GetDelta(); size.y *= isign; m_CurrentPad->SetDelta( size ); m_CurrentPad->SetDrillSize( m_Pad_Master.GetDrillSize() ); m_CurrentPad->SetDrillShape( m_Pad_Master.GetDrillShape() ); wxPoint offset = m_Pad_Master.GetOffset(); offset.y *= isign; m_CurrentPad->SetOffset( offset ); m_CurrentPad->SetDieLength( m_Pad_Master.GetDieLength() ); if( m_CurrentPad->GetLayerMask() != m_Pad_Master.GetLayerMask() ) { rastnestIsChanged = true; m_CurrentPad->SetLayerMask( m_Pad_Master.GetLayerMask() ); } if( m_isFlipped ) m_CurrentPad->SetLayerMask( ChangeSideMaskLayer( m_CurrentPad->GetLayerMask() ) ); m_CurrentPad->SetPadName( m_Pad_Master.GetPadName() ); if( m_CurrentPad->GetNetname() != m_Pad_Master.GetNetname() ) { if( m_Pad_Master.GetNetname().IsEmpty() ) { rastnestIsChanged = true; m_CurrentPad->SetNet( 0 ); m_CurrentPad->SetNetname( wxEmptyString ); } else { const NETINFO_ITEM* net = m_Parent->GetBoard()->FindNet( m_Pad_Master.GetNetname() ); if( net ) { rastnestIsChanged = true; m_CurrentPad->SetNetname( m_Pad_Master.GetNetname() ); m_CurrentPad->SetNet( net->GetNet() ); } else DisplayError( NULL, _( "Unknown netname, netname not changed" ) ); } } m_CurrentPad->SetLocalClearance( m_Pad_Master.GetLocalClearance() ); m_CurrentPad->SetLocalSolderMaskMargin( m_Pad_Master.GetLocalSolderMaskMargin() ); m_CurrentPad->SetLocalSolderPasteMargin( m_Pad_Master.GetLocalSolderPasteMargin() ); m_CurrentPad->SetLocalSolderPasteMarginRatio( m_Pad_Master.GetLocalSolderPasteMarginRatio() ); m_CurrentPad->SetZoneConnection( m_Pad_Master.GetZoneConnection() ); module->CalculateBoundingBox(); m_CurrentPad->DisplayInfo( m_Parent ); // redraw the area where the pad was m_Parent->GetCanvas()->RefreshDrawingRect( m_CurrentPad->GetBoundingBox() ); m_Parent->OnModify(); } EndModal( wxID_OK ); if( rastnestIsChanged ) // The net ratsnest must be recalculated m_Parent->GetBoard()->m_Status_Pcb = 0; } // Copy values from dialog to aPad parameters. // If aPromptOnError is true, and an incorrect value is detected, // user will be prompted for an error bool DIALOG_PAD_PROPERTIES::TransfertDataToPad( D_PAD* aPad, bool aPromptOnError ) { long padLayerMask; int internalUnits = m_Parent->GetInternalUnits(); wxString msg; int x, y; aPad->SetAttribute( CodeType[m_PadType->GetSelection()] ); aPad->SetShape( CodeShape[m_PadShape->GetSelection()] ); // Read pad clearances values: aPad->SetLocalClearance( ReturnValueFromTextCtrl( *m_NetClearanceValueCtrl, internalUnits ) ); aPad->SetLocalSolderMaskMargin( ReturnValueFromTextCtrl( *m_SolderMaskMarginCtrl, internalUnits ) ); aPad->SetLocalSolderPasteMargin( ReturnValueFromTextCtrl( *m_SolderPasteMarginCtrl, internalUnits ) ); double dtmp = 0.0; msg = m_SolderPasteMarginRatioCtrl->GetValue(); msg.ToDouble( &dtmp ); // A margin ratio of -50% means no paste on a pad, the ratio must be >= 50 % if( dtmp < -50 ) dtmp = -50; if( dtmp > +100 ) dtmp = +100; aPad->SetLocalSolderPasteMarginRatio( dtmp / 100 ); switch( m_ZoneConnectionChoice->GetSelection() ) { default: case 0: aPad->SetZoneConnection( UNDEFINED_CONNECTION ); break; case 1: aPad->SetZoneConnection( PAD_IN_ZONE ); break; case 2: aPad->SetZoneConnection( THERMAL_PAD ); break; case 3: aPad->SetZoneConnection( PAD_NOT_IN_ZONE ); break; } // Read pad position: x = ReturnValueFromTextCtrl( *m_PadPosition_X_Ctrl, internalUnits ); y = ReturnValueFromTextCtrl( *m_PadPosition_Y_Ctrl, internalUnits ); aPad->SetPosition( wxPoint( x, y ) ); aPad->SetPos0( wxPoint( x, y ) ); // Read pad drill: x = ReturnValueFromTextCtrl( *m_PadDrill_X_Ctrl, internalUnits ); y = ReturnValueFromTextCtrl( *m_PadDrill_Y_Ctrl, internalUnits ); if( m_DrillShapeCtrl->GetSelection() == 0 ) { aPad->SetDrillShape( PAD_CIRCLE ); y = x; } else aPad->SetDrillShape( PAD_OVAL ); aPad->SetDrillSize( wxSize( x, y ) ); // Read pad shape size: x = ReturnValueFromTextCtrl( *m_ShapeSize_X_Ctrl, internalUnits ); y = ReturnValueFromTextCtrl( *m_ShapeSize_Y_Ctrl, internalUnits ); if( aPad->GetShape() == PAD_CIRCLE ) y = x; aPad->SetSize( wxSize( x, y ) ); // Read pad length die aPad->SetDieLength( ReturnValueFromTextCtrl( *m_LengthDieCtrl, internalUnits ) ); // Read pad shape delta size: // m_DeltaSize.x or m_DeltaSize.y must be NULL. for a trapezoid. wxSize delta; if( m_radioBtnDeltaXdir->GetValue() ) delta.x = ReturnValueFromTextCtrl( *m_ShapeDelta_Ctrl, internalUnits ); else delta.y = ReturnValueFromTextCtrl( *m_ShapeDelta_Ctrl, internalUnits ); // Test bad values (be sure delta values are not too large) // remember DeltaSize.x is the Y size variation bool error = false; if( delta.x < 0 && delta.x <= -aPad->GetSize().y ) { delta.x = -aPad->GetSize().y + 2; error = true; } if( delta.x > 0 && delta.x >= aPad->GetSize().y ) { delta.x = aPad->GetSize().y - 2; error = true; } if( delta.y < 0 && delta.y <= -aPad->GetSize().x ) { delta.y = -aPad->GetSize().x + 2; error = true; } if( delta.y > 0 && delta.y >= aPad->GetSize().x ) { delta.y = aPad->GetSize().x - 2; error = true; } aPad->SetDelta( delta ); // Read pad shape offset: x = ReturnValueFromTextCtrl( *m_ShapeOffset_X_Ctrl, internalUnits ); y = ReturnValueFromTextCtrl( *m_ShapeOffset_Y_Ctrl, internalUnits ); aPad->SetOffset( wxPoint( x, y ) ); double orient_value = 0; msg = m_PadOrientCtrl->GetValue(); msg.ToDouble( &orient_value ); aPad->SetOrientation( orient_value ); msg = m_PadNumCtrl->GetValue().Left( 4 ); aPad->SetPadName( msg ); aPad->SetNetname( m_PadNetNameCtrl->GetValue() ); // Clear some values, according to the pad type and shape switch( aPad->GetShape() ) { case PAD_CIRCLE: aPad->SetOffset( wxPoint( 0, 0 ) ); aPad->SetDelta( wxSize( 0, 0 ) ); x = aPad->GetSize().x; aPad->SetSize( wxSize( x, x ) ); break; case PAD_RECT: aPad->SetDelta( wxSize( 0, 0 ) ); break; case PAD_OVAL: aPad->SetDelta( wxSize( 0, 0 ) ); break; case PAD_TRAPEZOID: break; default: ; } switch( aPad->GetAttribute() ) { case PAD_STANDARD: break; case PAD_CONN: case PAD_SMD: aPad->SetOffset( wxPoint( 0, 0 ) ); aPad->SetDrillSize( wxSize( 0, 0 ) ); break; case PAD_HOLE_NOT_PLATED: // Mechanical purpose only: // no offset, no net name, no pad name allowed aPad->SetOffset( wxPoint( 0, 0 ) ); aPad->SetPadName( wxEmptyString ); aPad->SetNetname( wxEmptyString ); break; default: DisplayError( NULL, wxT( "Error: unknown pad type" ) ); break; } padLayerMask = 0; switch( m_rbCopperLayersSel->GetSelection() ) { case 0: padLayerMask |= LAYER_FRONT; break; case 1: padLayerMask |= LAYER_BACK; break; case 2: padLayerMask |= ALL_CU_LAYERS; break; case 3: // No copper layers break; } if( m_PadLayerAdhCmp->GetValue() ) padLayerMask |= ADHESIVE_LAYER_FRONT; if( m_PadLayerAdhCu->GetValue() ) padLayerMask |= ADHESIVE_LAYER_BACK; if( m_PadLayerPateCmp->GetValue() ) padLayerMask |= SOLDERPASTE_LAYER_FRONT; if( m_PadLayerPateCu->GetValue() ) padLayerMask |= SOLDERPASTE_LAYER_BACK; if( m_PadLayerSilkCmp->GetValue() ) padLayerMask |= SILKSCREEN_LAYER_FRONT; if( m_PadLayerSilkCu->GetValue() ) padLayerMask |= SILKSCREEN_LAYER_BACK; if( m_PadLayerMaskCmp->GetValue() ) padLayerMask |= SOLDERMASK_LAYER_FRONT; if( m_PadLayerMaskCu->GetValue() ) padLayerMask |= SOLDERMASK_LAYER_BACK; if( m_PadLayerECO1->GetValue() ) padLayerMask |= ECO1_LAYER; if( m_PadLayerECO2->GetValue() ) padLayerMask |= ECO2_LAYER; if( m_PadLayerDraft->GetValue() ) padLayerMask |= DRAW_LAYER; aPad->SetLayerMask( padLayerMask ); // Test for incorrect values if( aPromptOnError ) { if( (aPad->GetSize().x < aPad->GetDrillSize().x) || (aPad->GetSize().y < aPad->GetDrillSize().y) ) { DisplayError( NULL, _( "Incorrect value for pad drill: pad drill bigger than pad size" ) ); return false; } int padlayers_mask = padLayerMask & (LAYER_BACK | LAYER_FRONT); if( padlayers_mask == 0 ) { if( aPad->GetDrillSize().x || aPad->GetDrillSize().y ) { msg = _( "Error: pad is not on a copper layer and has a hole" ); if( aPad->GetAttribute() == PAD_HOLE_NOT_PLATED ) { msg += wxT("\n"); msg += _( "For NPTH pad, set pad drill value to pad size value,\n\ if you do not want this pad plotted in gerber files"); } DisplayError( NULL, msg ); return false; } } if( aPad->GetSize().x / 2 <= ABS( aPad->GetOffset().x ) || aPad->GetSize().y / 2 <= ABS( aPad->GetOffset().y ) ) { DisplayError( NULL, _( "Incorrect value for pad offset" ) ); return false; } if( error ) { DisplayError( NULL, _( "Too large value for pad delta size" ) ); return false; } } return true; } // Called when a dimension has change. // Update the pad dimensions shown in the panel. void DIALOG_PAD_PROPERTIES::OnValuesChanged( wxCommandEvent& event ) { if( m_canUpdate ) { TransfertDataToPad( m_dummyPad ); m_panelShowPad->Refresh(); } } /*********************************************************************/ void DIALOG_PAD_PROPERTIES::OnCancelButtonClick( wxCommandEvent& event ) /*********************************************************************/ { EndModal( wxID_CANCEL ); }