mirror of
https://gitlab.com/kicad/code/kicad.git
synced 2025-09-14 18:23:15 +02:00
There has been a long standing (since the beginning of the project?) issue with sharing schematics between projects. It has been somewhat supported for complex hierarchies (a sheet shared multiple times in a single design) but it has not been well supported for simple hierarchies (the symbol references cannot be changed in the shared schematic). This issue has been resolved by moving all of the symbol instance sheet paths from the symbol definitions in the all of the project files and save all symbol path instances in the root sheet. This ensures that orphaned symbol instance paths do not accumulate in shared schematic files and that designs that reuse schematic in simple hierarchies can how have different references. It also allows the root schematic from one project to be uses as a sub-sheet in another project. When legacy schematics are loaded, all sheet and symbol UUIDs are converted from time stamps to true UUIDs. This is done to ensure there are no sheet path instance clashes between projects. That being said, there are no checks for this. It is assumed that the probability of UUID clashes is so low that it doesn't make sense to test for them.
736 lines
22 KiB
C++
736 lines
22 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2018-2020 KiCad Developers, see AUTHORS.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
|
|
*/
|
|
|
|
#include <kiway.h>
|
|
#include <kiway_player.h>
|
|
#include <dialog_shim.h>
|
|
#include <fields_grid_table.h>
|
|
#include <sch_base_frame.h>
|
|
#include <sch_field.h>
|
|
#include <sch_validators.h>
|
|
#include <validators.h>
|
|
#include <class_library.h>
|
|
#include <template_fieldnames.h>
|
|
#include <widgets/grid_icon_text_helpers.h>
|
|
#include <widgets/grid_text_button_helpers.h>
|
|
#include <wildcards_and_files_ext.h>
|
|
|
|
#include "eda_doc.h"
|
|
|
|
|
|
enum
|
|
{
|
|
MYID_SELECT_FOOTPRINT = 991, // must be within GRID_TRICKS' enum range
|
|
MYID_SHOW_DATASHEET
|
|
};
|
|
|
|
|
|
template <class T>
|
|
FIELDS_GRID_TABLE<T>::FIELDS_GRID_TABLE( DIALOG_SHIM* aDialog, SCH_BASE_FRAME* aFrame,
|
|
LIB_PART* aPart ) :
|
|
m_frame( aFrame ),
|
|
m_userUnits( aDialog->GetUserUnits() ),
|
|
m_parentType( SCH_COMPONENT_T ),
|
|
m_mandatoryFieldCount( MANDATORY_FIELDS ),
|
|
m_part( aPart ),
|
|
m_fieldNameValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), FIELD_NAME ),
|
|
m_referenceValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), REFERENCE ),
|
|
m_valueValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), VALUE ),
|
|
m_libIdValidator( LIB_ID::ID_PCB ),
|
|
m_urlValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), FIELD_VALUE ),
|
|
m_nonUrlValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), FIELD_VALUE ),
|
|
m_filepathValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), SHEETFILENAME )
|
|
{
|
|
initGrid( aDialog );
|
|
}
|
|
|
|
|
|
template <class T>
|
|
FIELDS_GRID_TABLE<T>::FIELDS_GRID_TABLE( DIALOG_SHIM* aDialog, SCH_BASE_FRAME* aFrame,
|
|
SCH_SHEET* aSheet ) :
|
|
m_frame( aFrame ),
|
|
m_userUnits( aDialog->GetUserUnits() ),
|
|
m_parentType( SCH_SHEET_T ),
|
|
m_mandatoryFieldCount( SHEET_MANDATORY_FIELDS ),
|
|
m_part( nullptr ),
|
|
m_fieldNameValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), FIELD_NAME ),
|
|
m_referenceValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), SHEETNAME_V ),
|
|
m_valueValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), VALUE ),
|
|
m_libIdValidator( LIB_ID::ID_PCB ),
|
|
m_urlValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), FIELD_VALUE ),
|
|
m_nonUrlValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), FIELD_VALUE ),
|
|
m_filepathValidator( aFrame->IsType( FRAME_SCH_LIB_EDITOR ), SHEETFILENAME_V )
|
|
{
|
|
initGrid( aDialog );
|
|
}
|
|
|
|
|
|
template <class T>
|
|
void FIELDS_GRID_TABLE<T>::initGrid( DIALOG_SHIM* aDialog )
|
|
{
|
|
// Build the various grid cell attributes.
|
|
// NOTE: validators and cellAttrs are member variables to get the destruction order
|
|
// right. wxGrid is VERY cranky about this.
|
|
|
|
m_readOnlyAttr = new wxGridCellAttr;
|
|
m_readOnlyAttr->SetReadOnly( true );
|
|
|
|
m_fieldNameAttr = new wxGridCellAttr;
|
|
GRID_CELL_TEXT_EDITOR* nameEditor = new GRID_CELL_TEXT_EDITOR();
|
|
nameEditor->SetValidator( m_fieldNameValidator );
|
|
m_fieldNameAttr->SetEditor( nameEditor );
|
|
|
|
m_referenceAttr = new wxGridCellAttr;
|
|
GRID_CELL_TEXT_EDITOR* referenceEditor = new GRID_CELL_TEXT_EDITOR();
|
|
referenceEditor->SetValidator( m_referenceValidator );
|
|
m_referenceAttr->SetEditor( referenceEditor );
|
|
|
|
m_valueAttr = new wxGridCellAttr;
|
|
GRID_CELL_TEXT_EDITOR* valueEditor = new GRID_CELL_TEXT_EDITOR();
|
|
valueEditor->SetValidator( m_valueValidator );
|
|
m_valueAttr->SetEditor( valueEditor );
|
|
|
|
m_footprintAttr = new wxGridCellAttr;
|
|
GRID_CELL_FOOTPRINT_ID_EDITOR* fpIdEditor = new GRID_CELL_FOOTPRINT_ID_EDITOR( aDialog );
|
|
fpIdEditor->SetValidator( m_libIdValidator );
|
|
m_footprintAttr->SetEditor( fpIdEditor );
|
|
|
|
m_urlAttr = new wxGridCellAttr;
|
|
GRID_CELL_URL_EDITOR* urlEditor = new GRID_CELL_URL_EDITOR( aDialog );
|
|
urlEditor->SetValidator( m_urlValidator );
|
|
m_urlAttr->SetEditor( urlEditor );
|
|
|
|
m_nonUrlAttr = new wxGridCellAttr;
|
|
GRID_CELL_TEXT_EDITOR* nonUrlEditor = new GRID_CELL_TEXT_EDITOR();
|
|
nonUrlEditor->SetValidator( m_nonUrlValidator );
|
|
m_nonUrlAttr->SetEditor( nonUrlEditor );
|
|
|
|
m_curdir = m_frame->Prj().GetProjectPath();
|
|
m_filepathAttr = new wxGridCellAttr;
|
|
|
|
// Create a wild card using wxFileDialog syntax.
|
|
wxString wildCard( _( "Schematic Files" ) );
|
|
std::vector<std::string> exts;
|
|
exts.push_back( KiCadSchematicFileExtension );
|
|
wildCard += AddFileExtListToFilter( exts );
|
|
|
|
GRID_CELL_PATH_EDITOR* filepathEditor =
|
|
new GRID_CELL_PATH_EDITOR( aDialog, &m_curdir, wildCard );
|
|
filepathEditor->SetValidator( m_filepathValidator );
|
|
m_filepathAttr->SetEditor( filepathEditor );
|
|
|
|
m_boolAttr = new wxGridCellAttr;
|
|
m_boolAttr->SetRenderer( new wxGridCellBoolRenderer() );
|
|
m_boolAttr->SetEditor( new wxGridCellBoolEditor() );
|
|
m_boolAttr->SetAlignment( wxALIGN_CENTER, wxALIGN_BOTTOM );
|
|
|
|
wxArrayString vAlignNames;
|
|
vAlignNames.Add( _( "Top" ) );
|
|
vAlignNames.Add( _( "Center" ) );
|
|
vAlignNames.Add( _( "Bottom" ) );
|
|
m_vAlignAttr = new wxGridCellAttr;
|
|
m_vAlignAttr->SetEditor( new wxGridCellChoiceEditor( vAlignNames ) );
|
|
m_vAlignAttr->SetAlignment( wxALIGN_CENTER, wxALIGN_BOTTOM );
|
|
|
|
wxArrayString hAlignNames;
|
|
hAlignNames.Add( _( "Left" ) );
|
|
hAlignNames.Add(_( "Center" ) );
|
|
hAlignNames.Add(_( "Right" ) );
|
|
m_hAlignAttr = new wxGridCellAttr;
|
|
m_hAlignAttr->SetEditor( new wxGridCellChoiceEditor( hAlignNames ) );
|
|
m_hAlignAttr->SetAlignment( wxALIGN_CENTER, wxALIGN_BOTTOM );
|
|
|
|
wxArrayString orientationNames;
|
|
orientationNames.Add( _( "Horizontal" ) );
|
|
orientationNames.Add(_( "Vertical" ) );
|
|
m_orientationAttr = new wxGridCellAttr;
|
|
m_orientationAttr->SetEditor( new wxGridCellChoiceEditor( orientationNames ) );
|
|
m_orientationAttr->SetAlignment( wxALIGN_CENTER, wxALIGN_BOTTOM );
|
|
}
|
|
|
|
|
|
template <class T>
|
|
FIELDS_GRID_TABLE<T>::~FIELDS_GRID_TABLE()
|
|
{
|
|
m_readOnlyAttr->DecRef();
|
|
m_fieldNameAttr->DecRef();
|
|
m_boolAttr->DecRef();
|
|
m_referenceAttr->DecRef();
|
|
m_valueAttr->DecRef();
|
|
m_footprintAttr->DecRef();
|
|
m_urlAttr->DecRef();
|
|
m_nonUrlAttr->DecRef();
|
|
m_filepathAttr->DecRef();
|
|
m_vAlignAttr->DecRef();
|
|
m_hAlignAttr->DecRef();
|
|
m_orientationAttr->DecRef();
|
|
}
|
|
|
|
|
|
template <class T>
|
|
wxString FIELDS_GRID_TABLE<T>::GetColLabelValue( int aCol )
|
|
{
|
|
switch( aCol )
|
|
{
|
|
case FDC_NAME: return _( "Name" );
|
|
case FDC_VALUE: return _( "Value" );
|
|
case FDC_SHOWN: return _( "Show" );
|
|
case FDC_H_ALIGN: return _( "H Align" );
|
|
case FDC_V_ALIGN: return _( "V Align" );
|
|
case FDC_ITALIC: return _( "Italic" );
|
|
case FDC_BOLD: return _( "Bold" );
|
|
case FDC_TEXT_SIZE: return _( "Text Size" );
|
|
case FDC_ORIENTATION: return _( "Orientation" );
|
|
case FDC_POSX: return _( "X Position" );
|
|
case FDC_POSY: return _( "Y Position" );
|
|
default: wxFAIL; return wxEmptyString;
|
|
}
|
|
}
|
|
|
|
|
|
template <class T>
|
|
bool FIELDS_GRID_TABLE<T>::CanGetValueAs( int aRow, int aCol, const wxString& aTypeName )
|
|
{
|
|
switch( aCol )
|
|
{
|
|
case FDC_NAME:
|
|
case FDC_VALUE:
|
|
case FDC_H_ALIGN:
|
|
case FDC_V_ALIGN:
|
|
case FDC_TEXT_SIZE:
|
|
case FDC_ORIENTATION:
|
|
case FDC_POSX:
|
|
case FDC_POSY:
|
|
return aTypeName == wxGRID_VALUE_STRING;
|
|
|
|
case FDC_SHOWN:
|
|
case FDC_ITALIC:
|
|
case FDC_BOLD:
|
|
return aTypeName == wxGRID_VALUE_BOOL;
|
|
|
|
default:
|
|
wxFAIL;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
template <class T>
|
|
bool FIELDS_GRID_TABLE<T>::CanSetValueAs( int aRow, int aCol, const wxString& aTypeName )
|
|
{
|
|
return CanGetValueAs( aRow, aCol, aTypeName );
|
|
}
|
|
|
|
|
|
template <class T>
|
|
wxGridCellAttr* FIELDS_GRID_TABLE<T>::GetAttr( int aRow, int aCol, wxGridCellAttr::wxAttrKind )
|
|
{
|
|
wxGridCellAttr* tmp;
|
|
|
|
// Only the VALUE and DATASHEET fields can be edited for inherited symbols.
|
|
bool rowIsReadOnly = m_part && m_part->IsAlias() && ( aRow == REFERENCE || aRow == FOOTPRINT );
|
|
|
|
switch( aCol )
|
|
{
|
|
case FDC_NAME:
|
|
if( aRow < m_mandatoryFieldCount || rowIsReadOnly )
|
|
{
|
|
tmp = m_fieldNameAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_fieldNameAttr->IncRef();
|
|
return m_fieldNameAttr;
|
|
}
|
|
|
|
case FDC_VALUE:
|
|
if( m_parentType == SCH_COMPONENT_T && aRow == REFERENCE )
|
|
{
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_referenceAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_referenceAttr->IncRef();
|
|
return m_referenceAttr;
|
|
}
|
|
}
|
|
else if( m_parentType == SCH_COMPONENT_T && aRow == VALUE )
|
|
{
|
|
// For power symbols, the value is not editable, because value and pin name must
|
|
// be the same and can be edited only in library editor.
|
|
if( ( m_part && m_part->IsPower() && !m_frame->IsType( FRAME_SCH_LIB_EDITOR ) )
|
|
|| rowIsReadOnly )
|
|
{
|
|
tmp = m_readOnlyAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_valueAttr->IncRef();
|
|
return m_valueAttr;
|
|
}
|
|
}
|
|
else if( m_parentType == SCH_COMPONENT_T && aRow == FOOTPRINT )
|
|
{
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_footprintAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_footprintAttr->IncRef();
|
|
return m_footprintAttr;
|
|
}
|
|
}
|
|
else if( m_parentType == SCH_COMPONENT_T && aRow == DATASHEET )
|
|
{
|
|
m_urlAttr->IncRef();
|
|
return m_urlAttr;
|
|
}
|
|
else if( m_parentType == SCH_SHEET_T && aRow == SHEETNAME )
|
|
{
|
|
m_referenceAttr->IncRef();
|
|
return m_referenceAttr;
|
|
}
|
|
else if( m_parentType == SCH_SHEET_T && aRow == SHEETFILENAME )
|
|
{
|
|
m_filepathAttr->IncRef();
|
|
return m_filepathAttr;
|
|
}
|
|
else
|
|
{
|
|
wxString fieldname = GetValue( aRow, FDC_NAME );
|
|
const TEMPLATE_FIELDNAME* templateFn = m_frame->GetTemplateFieldName( fieldname );
|
|
|
|
if( templateFn && templateFn->m_URL )
|
|
{
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_urlAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_urlAttr->IncRef();
|
|
return m_urlAttr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_nonUrlAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_nonUrlAttr->IncRef();
|
|
return m_nonUrlAttr;
|
|
}
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
case FDC_TEXT_SIZE:
|
|
case FDC_POSX:
|
|
case FDC_POSY:
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_readOnlyAttr->Clone();
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
case FDC_H_ALIGN:
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_hAlignAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_hAlignAttr->IncRef();
|
|
return m_hAlignAttr;
|
|
}
|
|
|
|
case FDC_V_ALIGN:
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_vAlignAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_vAlignAttr->IncRef();
|
|
return m_vAlignAttr;
|
|
}
|
|
|
|
case FDC_ORIENTATION:
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_orientationAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_orientationAttr->IncRef();
|
|
return m_orientationAttr;
|
|
}
|
|
|
|
case FDC_SHOWN:
|
|
case FDC_ITALIC:
|
|
case FDC_BOLD:
|
|
if( rowIsReadOnly )
|
|
{
|
|
tmp = m_boolAttr->Clone();
|
|
tmp->SetReadOnly( true );
|
|
tmp->SetTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
|
|
return tmp;
|
|
}
|
|
else
|
|
{
|
|
m_boolAttr->IncRef();
|
|
return m_boolAttr;
|
|
}
|
|
|
|
default:
|
|
wxFAIL;
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
|
|
template <class T>
|
|
wxString FIELDS_GRID_TABLE<T>::GetValue( int aRow, int aCol )
|
|
{
|
|
wxCHECK( aRow < GetNumberRows(), wxEmptyString );
|
|
const T& field = this->at( (size_t) aRow );
|
|
|
|
switch( aCol )
|
|
{
|
|
case FDC_NAME:
|
|
// Use default field name for mandatory fields, because they are translated
|
|
// according to the current locale
|
|
if( aRow < m_mandatoryFieldCount )
|
|
{
|
|
if( m_parentType == SCH_COMPONENT_T )
|
|
return TEMPLATE_FIELDNAME::GetDefaultFieldName( aRow );
|
|
else if( m_parentType == SCH_SHEET_T )
|
|
return SCH_SHEET::GetDefaultFieldName( aRow );
|
|
}
|
|
|
|
return field.GetName( false );
|
|
|
|
case FDC_VALUE:
|
|
return field.GetText();
|
|
|
|
case FDC_SHOWN:
|
|
return StringFromBool( field.IsVisible() );
|
|
|
|
case FDC_H_ALIGN:
|
|
switch ( field.GetHorizJustify() )
|
|
{
|
|
case GR_TEXT_HJUSTIFY_LEFT: return _( "Left" );
|
|
case GR_TEXT_HJUSTIFY_CENTER: return _( "Center" );
|
|
case GR_TEXT_HJUSTIFY_RIGHT: return _( "Right" );
|
|
}
|
|
|
|
break;
|
|
|
|
case FDC_V_ALIGN:
|
|
switch ( field.GetVertJustify() )
|
|
{
|
|
case GR_TEXT_VJUSTIFY_TOP: return _( "Top" );
|
|
case GR_TEXT_VJUSTIFY_CENTER: return _( "Center" );
|
|
case GR_TEXT_VJUSTIFY_BOTTOM: return _( "Bottom" );
|
|
}
|
|
|
|
break;
|
|
|
|
case FDC_ITALIC:
|
|
return StringFromBool( field.IsItalic() );
|
|
|
|
case FDC_BOLD:
|
|
return StringFromBool( field.IsBold() );
|
|
|
|
case FDC_TEXT_SIZE:
|
|
return StringFromValue( m_userUnits, field.GetTextSize().GetHeight(), true, true );
|
|
|
|
case FDC_ORIENTATION:
|
|
switch ( (int) field.GetTextAngle() )
|
|
{
|
|
case TEXT_ANGLE_HORIZ: return _( "Horizontal" );
|
|
case TEXT_ANGLE_VERT: return _( "Vertical" );
|
|
}
|
|
|
|
break;
|
|
|
|
case FDC_POSX:
|
|
return StringFromValue( m_userUnits, field.GetTextPos().x, true );
|
|
|
|
case FDC_POSY:
|
|
return StringFromValue( m_userUnits, field.GetTextPos().y, true );
|
|
|
|
default:
|
|
// we can't assert here because wxWidgets sometimes calls this without checking
|
|
// the column type when trying to see if there's an overflow
|
|
break;
|
|
}
|
|
|
|
return wxT( "bad wxWidgets!" );
|
|
}
|
|
|
|
|
|
template <class T>
|
|
bool FIELDS_GRID_TABLE<T>::GetValueAsBool( int aRow, int aCol )
|
|
{
|
|
wxCHECK( aRow < GetNumberRows(), false );
|
|
const T& field = this->at( (size_t) aRow );
|
|
|
|
switch( aCol )
|
|
{
|
|
case FDC_SHOWN: return field.IsVisible();
|
|
case FDC_ITALIC: return field.IsItalic();
|
|
case FDC_BOLD: return field.IsBold();
|
|
default:
|
|
wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a bool value" ), aCol ) );
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
template <class T>
|
|
void FIELDS_GRID_TABLE<T>::SetValue( int aRow, int aCol, const wxString &aValue )
|
|
{
|
|
wxCHECK( aRow < GetNumberRows(), /*void*/ );
|
|
T& field = this->at( (size_t) aRow );
|
|
wxPoint pos;
|
|
|
|
switch( aCol )
|
|
{
|
|
case FDC_NAME:
|
|
field.SetName( aValue );
|
|
break;
|
|
|
|
case FDC_VALUE:
|
|
field.SetText( aValue );
|
|
break;
|
|
|
|
case FDC_SHOWN:
|
|
field.SetVisible( BoolFromString( aValue ) );
|
|
break;
|
|
|
|
case FDC_H_ALIGN:
|
|
if( aValue == _( "Left" ) )
|
|
field.SetHorizJustify( GR_TEXT_HJUSTIFY_LEFT );
|
|
else if( aValue == _( "Center" ) )
|
|
field.SetHorizJustify( GR_TEXT_HJUSTIFY_CENTER );
|
|
else if( aValue == _( "Right" ) )
|
|
field.SetHorizJustify( GR_TEXT_HJUSTIFY_RIGHT );
|
|
else
|
|
wxFAIL_MSG( wxT( "unknown horizontal alignment: " ) + aValue );
|
|
break;
|
|
|
|
case FDC_V_ALIGN:
|
|
if( aValue == _( "Top" ) )
|
|
field.SetVertJustify( GR_TEXT_VJUSTIFY_TOP );
|
|
else if( aValue == _( "Center" ) )
|
|
field.SetVertJustify( GR_TEXT_VJUSTIFY_CENTER );
|
|
else if( aValue == _( "Bottom" ) )
|
|
field.SetVertJustify( GR_TEXT_VJUSTIFY_BOTTOM );
|
|
else
|
|
wxFAIL_MSG( wxT( "unknown vertical alignment: " ) + aValue);
|
|
break;
|
|
|
|
case FDC_ITALIC:
|
|
field.SetItalic( BoolFromString( aValue ) );
|
|
break;
|
|
|
|
case FDC_BOLD:
|
|
field.SetBold( BoolFromString( aValue ) );
|
|
break;
|
|
|
|
case FDC_TEXT_SIZE:
|
|
field.SetTextSize( wxSize( ValueFromString( m_userUnits, aValue ),
|
|
ValueFromString( m_userUnits, aValue ) ) );
|
|
break;
|
|
|
|
case FDC_ORIENTATION:
|
|
if( aValue == _( "Horizontal" ) )
|
|
field.SetTextAngle( TEXT_ANGLE_HORIZ );
|
|
else if( aValue == _( "Vertical" ) )
|
|
field.SetTextAngle( TEXT_ANGLE_VERT );
|
|
else
|
|
wxFAIL_MSG( wxT( "unknown orientation: " ) + aValue );
|
|
break;
|
|
|
|
case FDC_POSX:
|
|
case FDC_POSY:
|
|
pos = field.GetTextPos();
|
|
if( aCol == FDC_POSX )
|
|
pos.x = ValueFromString( m_userUnits, aValue );
|
|
else
|
|
pos.y = ValueFromString( m_userUnits, aValue );
|
|
field.SetTextPos( pos );
|
|
break;
|
|
|
|
default:
|
|
wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a string value" ), aCol ) );
|
|
break;
|
|
}
|
|
|
|
GetView()->Refresh();
|
|
}
|
|
|
|
|
|
template <class T>
|
|
void FIELDS_GRID_TABLE<T>::SetValueAsBool( int aRow, int aCol, bool aValue )
|
|
{
|
|
wxCHECK( aRow < GetNumberRows(), /*void*/ );
|
|
T& field = this->at( (size_t) aRow );
|
|
|
|
switch( aCol )
|
|
{
|
|
case FDC_SHOWN:
|
|
field.SetVisible( aValue );
|
|
break;
|
|
case FDC_ITALIC:
|
|
field.SetItalic( aValue );
|
|
break;
|
|
case FDC_BOLD:
|
|
field.SetBold( aValue );
|
|
break;
|
|
default:
|
|
wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a bool value" ), aCol ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// Explicit Instantiations
|
|
|
|
template class FIELDS_GRID_TABLE<SCH_FIELD>;
|
|
template class FIELDS_GRID_TABLE<LIB_FIELD>;
|
|
|
|
|
|
void FIELDS_GRID_TRICKS::showPopupMenu( wxMenu& menu )
|
|
{
|
|
if( m_grid->GetGridCursorRow() == FOOTPRINT && m_grid->GetGridCursorCol() == FDC_VALUE )
|
|
{
|
|
menu.Append( MYID_SELECT_FOOTPRINT, _( "Select Footprint..." ),
|
|
_( "Browse for footprint" ) );
|
|
menu.AppendSeparator();
|
|
}
|
|
else if( m_grid->GetGridCursorRow() == DATASHEET && m_grid->GetGridCursorCol() == FDC_VALUE )
|
|
{
|
|
menu.Append( MYID_SHOW_DATASHEET, _( "Show Datasheet" ),
|
|
_( "Show datasheet in browser" ) );
|
|
menu.AppendSeparator();
|
|
}
|
|
|
|
GRID_TRICKS::showPopupMenu( menu );
|
|
}
|
|
|
|
|
|
void FIELDS_GRID_TRICKS::doPopupSelection( wxCommandEvent& event )
|
|
{
|
|
if( event.GetId() == MYID_SELECT_FOOTPRINT )
|
|
{
|
|
// pick a footprint using the footprint picker.
|
|
wxString fpid = m_grid->GetCellValue( FOOTPRINT, FDC_VALUE );
|
|
KIWAY_PLAYER* frame = m_dlg->Kiway().Player( FRAME_FOOTPRINT_VIEWER_MODAL, true, m_dlg );
|
|
|
|
if( frame->ShowModal( &fpid, m_dlg ) )
|
|
m_grid->SetCellValue( FOOTPRINT, FDC_VALUE, fpid );
|
|
|
|
frame->Destroy();
|
|
}
|
|
else if (event.GetId() == MYID_SHOW_DATASHEET )
|
|
{
|
|
wxString datasheet_uri = m_grid->GetCellValue( DATASHEET, FDC_VALUE );
|
|
GetAssociatedDocument( m_dlg, datasheet_uri, &m_dlg->Prj() );
|
|
}
|
|
else
|
|
{
|
|
GRID_TRICKS::doPopupSelection( event );
|
|
}
|
|
}
|
|
|
|
|
|
template <class T>
|
|
wxString FIELDS_GRID_TABLE<T>::StringFromBool( bool aValue )
|
|
{
|
|
if( aValue )
|
|
return wxT( "1" );
|
|
else
|
|
return wxT( "0" );
|
|
}
|
|
|
|
|
|
template <class T>
|
|
bool FIELDS_GRID_TABLE<T>::BoolFromString( wxString aValue )
|
|
{
|
|
if( aValue == "1" )
|
|
{
|
|
return true;
|
|
}
|
|
else if( aValue == "0" )
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
wxFAIL_MSG( wxString::Format( "string \"%s\" can't be converted to boolean "
|
|
"correctly, it will have been perceived as FALSE", aValue ) );
|
|
return false;
|
|
}
|
|
}
|