2011-08-05 19:53:42 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KICAD, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2011 jean-pierre.charras
|
|
|
|
* Copyright (C) 1992-2011 Kicad Developers, see change_log.txt for contributors.
|
|
|
|
*
|
|
|
|
* 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, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <wx/wx.h>
|
|
|
|
#include <wx/config.h>
|
|
|
|
#include <wx/filename.h>
|
2011-08-05 19:53:42 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <pcb_calculator_frame_base.h>
|
2011-08-05 19:53:42 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <pcb_calculator.h>
|
|
|
|
#include <UnitSelector.h>
|
2011-08-05 19:53:42 +00:00
|
|
|
|
|
|
|
extern double ReturnDoubleFromString( const wxString& TextValue );
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* these values come from QucsStudio ( by Michael Margraf )
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Display a selection of usual Er, TanD, Rho values
|
|
|
|
// format is <value><space><comment>
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function OnEpsilonR_Button
|
|
|
|
* Shows a list of current relative dielectric constant(Er)
|
|
|
|
* and set the selected value in main dialog frame
|
|
|
|
*/
|
|
|
|
void PCB_CALCULATOR_FRAME::OnTranslineEpsilonR_Button( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
wxArrayString list;
|
|
|
|
|
|
|
|
// EpsilonR ( relative dielectric constant) list
|
|
|
|
list.Add( wxT( "4.5 FR4" ) );
|
|
|
|
list.Add( wxT( "9.8 alumina (Al2O3)" ) );
|
|
|
|
list.Add( wxT( "3.78 fused quartz" ) );
|
|
|
|
list.Add( wxT( "3.38 RO4003" ) );
|
|
|
|
list.Add( wxT( "2.2 RT/duroid 5880" ) );
|
|
|
|
list.Add( wxT( "10.2 RT/duroid 6010LM" ) );
|
|
|
|
list.Add( wxT( "2.1 teflon (PTFE)" ) );
|
|
|
|
list.Add( wxT( "4.0 PVC" ) );
|
|
|
|
list.Add( wxT( "2.3 PE" ) );
|
|
|
|
list.Add( wxT( "6.6 beryllia (BeO)" ) );
|
|
|
|
list.Add( wxT( "8.7 aluminum nitride" ) );
|
|
|
|
list.Add( wxT( "11.9 silicon" ) );
|
|
|
|
list.Add( wxT( "12.9 GaAs" ) );
|
|
|
|
|
|
|
|
wxString value = wxGetSingleChoice( wxEmptyString,
|
|
|
|
_("Relative Dielectric Constants"), list).BeforeFirst( ' ' );
|
|
|
|
if( ! value.IsEmpty() )
|
|
|
|
m_Value_EpsilonR->SetValue( value );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function OnTanD_Button
|
|
|
|
* Shows a list of current dielectric loss factor (tangent delta)
|
|
|
|
* and set the selected value in main dialog frame
|
|
|
|
*/
|
|
|
|
void PCB_CALCULATOR_FRAME::OnTranslineTanD_Button( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
wxArrayString list;
|
|
|
|
|
|
|
|
// List of current dielectric loss factor (tangent delta)
|
|
|
|
list.Clear();
|
|
|
|
list.Add( wxT( "2e-2 FR4 @ 1GHz" ) );
|
|
|
|
list.Add( wxT( "3e-4 beryllia @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "2e-4 aluminia (Al2O3) @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "1e-4 fused quartz @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "2e-3 RO4003 @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "9e-4 RT/duroid 5880 @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "2e-4 teflon (PTFE) @ 1MHz" ) );
|
|
|
|
list.Add( wxT( "5e-2 PVC @ 1MHz" ) );
|
|
|
|
list.Add( wxT( "2e-4 PE @ 1MHz" ) );
|
|
|
|
list.Add( wxT( "1e-3 aluminum nitride @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "0.015 silicon @ 10GHz" ) );
|
|
|
|
list.Add( wxT( "0.002 GaAs @ 10GHz" ) );
|
|
|
|
|
|
|
|
wxString value = wxGetSingleChoice( wxEmptyString,
|
|
|
|
_("Dielectric Loss Factor"), list).BeforeFirst( ' ' );
|
|
|
|
if( ! value.IsEmpty() )
|
|
|
|
m_Value_TanD->SetValue( value );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function OnTranslineRho_Button
|
|
|
|
* Shows a list of current Specific resistance list (rho)
|
|
|
|
* and set the selected value in main dialog frame
|
|
|
|
*/
|
|
|
|
void PCB_CALCULATOR_FRAME::OnTranslineRho_Button( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
wxArrayString list;
|
|
|
|
|
|
|
|
// Specific resistance list in ohms*meters (rho):
|
|
|
|
list.Clear();
|
|
|
|
list.Add( wxT( "2.4e-8 gold" ) );
|
|
|
|
list.Add( wxT( "1.72e-8 copper" ) );
|
|
|
|
list.Add( wxT( "1.62e-8 silver" ) );
|
|
|
|
list.Add( wxT( "12.4e-8 tin" ) );
|
|
|
|
list.Add( wxT( "10.5e-8 platinum" ) );
|
|
|
|
list.Add( wxT( "2.62e-8 aluminum" ) );
|
|
|
|
list.Add( wxT( "6.9e-8 nickel" ) );
|
|
|
|
list.Add( wxT( "3.9e-8 brass (66Cu 34Zn)" ) );
|
|
|
|
list.Add( wxT( "9.71e-8 iron" ) );
|
|
|
|
list.Add( wxT( "6.0e-8 zinc" ) );
|
|
|
|
|
|
|
|
wxString value = wxGetSingleChoice( wxEmptyString,
|
|
|
|
_("Specific Resistance"), list).BeforeFirst( ' ' );
|
|
|
|
if( ! value.IsEmpty() )
|
|
|
|
m_Value_Rho->SetValue( value );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Minor helper struct to handle dialog items for a given parameter
|
|
|
|
struct DLG_PRM_DATA
|
|
|
|
{
|
|
|
|
wxStaticText * name;
|
|
|
|
wxTextCtrl * value;
|
|
|
|
UNIT_SELECTOR * unit;
|
|
|
|
};
|
|
|
|
/**
|
|
|
|
* Function TranslineTypeSelection
|
|
|
|
* Must be called after selection of a new transline.
|
|
|
|
* Update all values, labels and tool tips of parameters needed
|
|
|
|
* by the new transline
|
|
|
|
* Irrelevant parameters texts are blanked.
|
|
|
|
* @param aType = the transline_type_id of the new selected transline
|
|
|
|
*/
|
|
|
|
void PCB_CALCULATOR_FRAME::TranslineTypeSelection( enum transline_type_id aType )
|
|
|
|
{
|
|
|
|
wxString msg;
|
|
|
|
#define DOUBLE_TO_CTLR( dlg_item, value ) { msg.Printf( wxT( "%g" ), value );\
|
|
|
|
dlg_item->SetValue( msg ); }
|
|
|
|
m_currTransLineType = aType;
|
|
|
|
|
|
|
|
if( (m_currTransLineType < start_of_list_type )
|
|
|
|
|| ( m_currTransLineType >= end_of_list_type ) )
|
|
|
|
m_currTransLineType = default_type;
|
|
|
|
|
|
|
|
TRANSLINE_IDENT* tr_ident = m_transline_list[m_currTransLineType];
|
|
|
|
m_currTransLine = tr_ident->m_TLine;
|
|
|
|
|
|
|
|
m_radioBtnPrm1->Show( tr_ident->m_HasPrmSelection );
|
|
|
|
m_radioBtnPrm2->Show( tr_ident->m_HasPrmSelection );
|
|
|
|
|
|
|
|
// Setup messages
|
|
|
|
wxStaticText* left_msg_list[] =
|
|
|
|
{
|
|
|
|
m_left_message1, m_left_message2, m_left_message3,
|
|
|
|
m_left_message4, m_left_message5, m_left_message6,
|
|
|
|
m_left_message7, NULL
|
|
|
|
};
|
|
|
|
wxStaticText* msg_list[] =
|
|
|
|
{
|
|
|
|
m_Message1, m_Message2, m_Message3,
|
|
|
|
m_Message4, m_Message5, m_Message6,
|
|
|
|
m_Message7, NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
unsigned ii = 0;
|
|
|
|
for( ; ii < tr_ident->m_Messages.GetCount(); ii++ )
|
|
|
|
{
|
|
|
|
if( left_msg_list[ii] == NULL )
|
|
|
|
break;
|
|
|
|
left_msg_list[ii]->SetLabel( tr_ident->m_Messages[ii] );
|
|
|
|
msg_list[ii]->SetLabel( wxEmptyString );
|
|
|
|
}
|
|
|
|
|
|
|
|
while( left_msg_list[ii] )
|
|
|
|
{
|
|
|
|
left_msg_list[ii]->SetLabel( wxEmptyString );
|
|
|
|
msg_list[ii]->SetLabel( wxEmptyString );
|
|
|
|
ii++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Init parameters dialog items
|
|
|
|
struct DLG_PRM_DATA substrateprms[] =
|
|
|
|
{
|
|
|
|
{ m_EpsilonR_label,m_Value_EpsilonR, NULL },
|
|
|
|
{ m_TanD_label,m_Value_TanD, NULL },
|
|
|
|
{ m_Rho_label, m_Value_Rho, NULL },
|
|
|
|
{ m_substrate_prm4_label,m_Substrate_prm4_Value, m_SubsPrm4_choiceUnit },
|
|
|
|
{ m_substrate_prm5_label,m_Substrate_prm5_Value, m_SubsPrm5_choiceUnit },
|
|
|
|
{ m_substrate_prm6_label,m_Substrate_prm6_Value, m_SubsPrm6_choiceUnit },
|
|
|
|
{ m_substrate_prm7_label,m_Substrate_prm7_Value, m_SubsPrm7_choiceUnit },
|
|
|
|
{ m_substrate_prm8_label,m_Substrate_prm8_Value, m_SubsPrm8_choiceUnit },
|
|
|
|
{ m_substrate_prm9_label,m_Substrate_prm9_Value, m_SubsPrm9_choiceUnit }
|
|
|
|
};
|
|
|
|
#define substrateprms_cnt (sizeof(substrateprms)/sizeof(substrateprms[0]))
|
|
|
|
|
|
|
|
struct DLG_PRM_DATA physprms[] =
|
|
|
|
{
|
|
|
|
{ m_phys_prm1_label,m_Phys_prm1_Value,m_choiceUnit_Param1 },
|
|
|
|
{ m_phys_prm2_label,m_Phys_prm2_Value,m_choiceUnit_Param2 },
|
|
|
|
{ m_phys_prm3_label,m_Phys_prm3_Value,m_choiceUnit_Param3 }
|
|
|
|
};
|
|
|
|
#define physprms_cnt (sizeof(physprms)/sizeof(physprms[0]))
|
|
|
|
|
|
|
|
struct DLG_PRM_DATA elecprms[] =
|
|
|
|
{
|
|
|
|
{ m_elec_prm1_label,m_Elec_prm1_Value, m_choiceUnit_ElecPrm1 },
|
|
|
|
{ m_elec_prm2_label,m_Elec_prm2_Value, m_choiceUnit_ElecPrm2 },
|
|
|
|
{ m_elec_prm3_label,m_Elec_prm3_Value, m_choiceUnit_ElecPrm3 }
|
|
|
|
};
|
|
|
|
#define elecprms_cnt (sizeof(elecprms)/sizeof(elecprms[0]))
|
|
|
|
|
|
|
|
struct DLG_PRM_DATA frequencyprms[] =
|
|
|
|
{
|
|
|
|
{ m_Frequency_label,m_Value_Frequency_Ctrl, m_choiceUnit_Frequency }
|
|
|
|
};
|
|
|
|
#define frequencyprms_cnt (sizeof(frequencyprms)/sizeof(frequencyprms[0]))
|
|
|
|
|
|
|
|
unsigned idxsubs = 0;
|
|
|
|
unsigned idxphys = 0;
|
|
|
|
unsigned idxelec = 0;
|
|
|
|
unsigned idxfreq = 0;
|
|
|
|
|
|
|
|
for( unsigned ii = 0; ii < tr_ident->GetPrmsCount(); ii++ )
|
|
|
|
{
|
|
|
|
TRANSLINE_PRM* prm = tr_ident->GetPrm( ii );
|
|
|
|
struct DLG_PRM_DATA * data = NULL;
|
|
|
|
switch( prm->m_Type )
|
|
|
|
{
|
|
|
|
case PRM_TYPE_SUBS:
|
|
|
|
wxASSERT( idxsubs < substrateprms_cnt );
|
|
|
|
data = &substrateprms[idxsubs];
|
|
|
|
idxsubs++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PRM_TYPE_PHYS:
|
|
|
|
wxASSERT( idxphys < physprms_cnt );
|
|
|
|
data = &physprms[idxphys];
|
|
|
|
idxphys++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PRM_TYPE_ELEC:
|
|
|
|
wxASSERT( idxelec < elecprms_cnt );
|
|
|
|
data = &elecprms[idxelec];
|
|
|
|
idxelec++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PRM_TYPE_FREQUENCY:
|
|
|
|
wxASSERT( idxfreq < frequencyprms_cnt );
|
|
|
|
data = &frequencyprms[idxfreq];
|
|
|
|
idxfreq++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
wxASSERT ( data );
|
|
|
|
data->name->SetToolTip( prm->m_ToolTip );
|
|
|
|
data->name->SetLabel( prm->m_Label );
|
|
|
|
prm->m_ValueCtrl = data->value;
|
|
|
|
if( prm->m_Id != DUMMY_PRM )
|
|
|
|
{
|
|
|
|
DOUBLE_TO_CTLR( data->value, prm->m_Value );
|
|
|
|
data->value->Enable( true );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
data->value->SetValue( wxEmptyString );
|
|
|
|
data->value->Enable( false );
|
|
|
|
}
|
|
|
|
if( prm->m_ConvUnit )
|
|
|
|
prm->m_UnitCtrl = data->unit;
|
|
|
|
if( data->unit )
|
|
|
|
{
|
|
|
|
data->unit->Show( prm->m_ConvUnit );
|
|
|
|
data->unit->Enable( prm->m_ConvUnit );
|
|
|
|
data->unit->SetSelection( prm->m_UnitSelection );
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear all unused params
|
|
|
|
for( ; idxsubs < substrateprms_cnt; idxsubs++ )
|
|
|
|
{
|
|
|
|
substrateprms[idxsubs].name->SetLabel(wxEmptyString);
|
|
|
|
substrateprms[idxsubs].name->SetToolTip(wxEmptyString);
|
|
|
|
substrateprms[idxsubs].value->SetValue(wxEmptyString);
|
|
|
|
substrateprms[idxsubs].value->Enable( false );
|
|
|
|
if( substrateprms[idxsubs].unit)
|
|
|
|
{
|
|
|
|
substrateprms[idxsubs].unit->Show( false );
|
|
|
|
substrateprms[idxsubs].unit->Enable( false );
|
|
|
|
substrateprms[idxsubs].unit->SetSelection( 0 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( ; idxphys < physprms_cnt; idxphys++ )
|
|
|
|
{
|
|
|
|
physprms[idxphys].name->SetLabel(wxEmptyString);
|
|
|
|
physprms[idxphys].name->SetToolTip(wxEmptyString);
|
|
|
|
physprms[idxphys].value->SetValue(wxEmptyString);
|
|
|
|
physprms[idxphys].value->Enable( false );
|
|
|
|
if( physprms[idxphys].unit)
|
|
|
|
{
|
|
|
|
physprms[idxphys].unit->Show( false );
|
|
|
|
physprms[idxphys].unit->Enable( false );
|
|
|
|
physprms[idxphys].unit->SetSelection( 0 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( ; idxelec < elecprms_cnt; idxelec++)
|
|
|
|
{
|
|
|
|
elecprms[idxelec].name->SetLabel(wxEmptyString);
|
|
|
|
elecprms[idxelec].name->SetToolTip(wxEmptyString);
|
|
|
|
elecprms[idxelec].value->SetValue(wxEmptyString);
|
|
|
|
elecprms[idxelec].value->Enable( false );
|
|
|
|
if( elecprms[idxelec].unit)
|
|
|
|
{
|
|
|
|
elecprms[idxelec].unit->Show( false );
|
|
|
|
elecprms[idxelec].unit->Enable( false );
|
|
|
|
elecprms[idxelec].unit->SetSelection( 0 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( ; idxfreq < frequencyprms_cnt; idxfreq++ )
|
|
|
|
{
|
|
|
|
frequencyprms[idxfreq].name->SetLabel(wxEmptyString);
|
|
|
|
frequencyprms[idxfreq].name->SetToolTip(wxEmptyString);
|
|
|
|
frequencyprms[idxfreq].value->SetValue(wxEmptyString);
|
|
|
|
frequencyprms[idxfreq].value->Enable( false );
|
|
|
|
if( frequencyprms[idxfreq].unit )
|
|
|
|
{
|
|
|
|
frequencyprms[idxfreq].unit->Show( false );
|
|
|
|
frequencyprms[idxfreq].unit->Enable( false );
|
|
|
|
frequencyprms[idxfreq].unit->SetSelection( 0 );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function TransfDlgDataToTranslineParams
|
|
|
|
* Read values entered in dialog frame, and transfert these
|
|
|
|
* values in current transline parameters, converted in normalized units
|
|
|
|
*/
|
|
|
|
void PCB_CALCULATOR_FRAME::TransfDlgDataToTranslineParams()
|
|
|
|
{
|
|
|
|
TRANSLINE_IDENT* tr_ident = m_transline_list[m_currTransLineType];
|
|
|
|
for( unsigned ii = 0; ii < tr_ident->GetPrmsCount(); ii++ )
|
|
|
|
{
|
|
|
|
TRANSLINE_PRM* prm = tr_ident->GetPrm( ii );
|
|
|
|
wxTextCtrl * value_ctrl = (wxTextCtrl * ) prm->m_ValueCtrl;
|
|
|
|
wxString value_txt = value_ctrl->GetValue();
|
|
|
|
double value = ReturnDoubleFromString(value_txt);
|
|
|
|
prm->m_Value = value;
|
|
|
|
UNIT_SELECTOR * unit_ctrl = (UNIT_SELECTOR * ) prm->m_UnitCtrl;
|
|
|
|
if( unit_ctrl )
|
|
|
|
{
|
|
|
|
prm->m_UnitSelection = unit_ctrl->GetSelection();
|
|
|
|
value *= unit_ctrl->GetUnitScale();
|
|
|
|
}
|
|
|
|
prm->m_NormalizedValue = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function OnTranslineSelection
|
|
|
|
* Called on new transmission line selection
|
|
|
|
*/
|
|
|
|
void PCB_CALCULATOR_FRAME::OnTranslineSelection( wxCommandEvent& event )
|
|
|
|
{
|
|
|
|
enum transline_type_id id = (enum transline_type_id) event.GetSelection();
|
|
|
|
|
|
|
|
TranslineTypeSelection( id );
|
|
|
|
|
|
|
|
// Texts and units choice widgets can have their size modified:
|
|
|
|
// The new size must be taken in account
|
|
|
|
m_panelTransline->GetSizer()->Layout();
|
|
|
|
m_panelTransline->Refresh();
|
|
|
|
}
|
|
|
|
|