KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
template_fieldnames.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2010 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
5 * Copyright The KiCad Developers, see AUTHORS.TXT for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
25#include "template_fieldnames.h"
26
27#include <mutex>
28
29#include <template_fieldnames_lexer.h>
30#include <string_utils.h>
31
32using namespace TFIELD_T;
33
34// N.B. Do not change these values without transitioning the file format
35#define REFERENCE_CANONICAL "Reference"
36#define VALUE_CANONICAL "Value"
37#define FOOTPRINT_CANONICAL "Footprint"
38#define DATASHEET_CANONICAL "Datasheet"
39#define DESCRIPTION_CANONICAL "Description"
40#define SHEET_NAME_CANONICAL "Sheetname"
41#define SHEET_FILE_CANONICAL "Sheetfile"
42#define INTERSHEET_REFS_CANONICAL "Intersheetrefs"
43#define USER_FIELD_CANONICAL "Field%d"
44
53
54
55wxString GetDefaultFieldName( FIELD_T aFieldId, bool aTranslateForHI )
56{
57 if( !aTranslateForHI )
58 {
59 switch( aFieldId )
60 {
61 case FIELD_T::REFERENCE: return s_CanonicalReference; // The symbol reference, R1, C1, etc.
62 case FIELD_T::VALUE: return s_CanonicalValue; // The symbol value
63 case FIELD_T::FOOTPRINT: return s_CanonicalFootprint; // The footprint for use with Pcbnew
64 case FIELD_T::DATASHEET: return s_CanonicalDatasheet; // Link to a datasheet for symbol
65 case FIELD_T::DESCRIPTION: return s_CanonicalDescription; // The symbol description
69 default: return GetUserFieldName( 42, aTranslateForHI );
70 }
71 }
72 else
73 {
74 switch( aFieldId )
75 {
76 case FIELD_T::REFERENCE: return _( REFERENCE_CANONICAL ); // The symbol reference, R1, C1, etc.
77 case FIELD_T::VALUE: return _( VALUE_CANONICAL ); // The symbol value
78 case FIELD_T::FOOTPRINT: return _( FOOTPRINT_CANONICAL ); // The footprint for use with Pcbnew
79 case FIELD_T::DATASHEET: return _( DATASHEET_CANONICAL ); // Link to a datasheet for symbol
80 case FIELD_T::DESCRIPTION: return _( DESCRIPTION_CANONICAL ); // The symbol description
84 default: return GetUserFieldName( 42, aTranslateForHI );
85 }
86 }
87}
88
89
90wxString GetUserFieldName( int aFieldNdx, bool aTranslateForHI )
91{
92 if( !aTranslateForHI )
93 return wxString::Format( wxS( USER_FIELD_CANONICAL ), aFieldNdx );
94 else
95 return wxString::Format( _( USER_FIELD_CANONICAL ), aFieldNdx );
96}
97
98
100{
101 out->Print( "(field (name %s)", out->Quotew( m_Name ).c_str() );
102
103 if( m_Visible )
104 out->Print( " visible" );
105
106 if( m_URL )
107 out->Print( " url" );
108
109 out->Print( ")" );
110}
111
112
113void TEMPLATE_FIELDNAME::Parse( TEMPLATE_FIELDNAMES_LEXER* in )
114{
115 T tok;
116
117 in->NeedLEFT(); // begin (name ...)
118
119 if( ( tok = in->NextTok() ) != T_name )
120 in->Expecting( T_name );
121
122 in->NeedSYMBOLorNUMBER();
123
124 m_Name = From_UTF8( in->CurText() );
125
126 in->NeedRIGHT(); // end (name ...)
127
128 while( (tok = in->NextTok() ) != T_RIGHT && tok != T_EOF )
129 {
130 // "visible" has no '(' prefix, "value" does, so T_LEFT is optional.
131 if( tok == T_LEFT )
132 tok = in->NextTok();
133
134 switch( tok )
135 {
136 case T_value:
137 // older format; silently skip
138 in->NeedSYMBOLorNUMBER();
139 in->NeedRIGHT();
140 break;
141
142 case T_visible:
143 m_Visible = true;
144 break;
145
146 case T_url:
147 m_URL = true;
148 break;
149
150 default:
151 in->Expecting( "value|url|visible" );
152 break;
153 }
154 }
155}
156
157
158void TEMPLATES::Format( OUTPUTFORMATTER* out, bool aGlobal ) const
159{
160 // We'll keep this general, and include the \n, even though the only known
161 // use at this time will not want the newlines or the indentation.
162 out->Print( "(templatefields" );
163
164 const std::vector<TEMPLATE_FIELDNAME>& source = aGlobal ? m_globals : m_project;
165
166 for( const TEMPLATE_FIELDNAME& temp : source )
167 {
168 if( !temp.m_Name.IsEmpty() )
169 temp.Format( out );
170 }
171
172 out->Print( ")" );
173}
174
175
176void TEMPLATES::parse( TEMPLATE_FIELDNAMES_LEXER* in, bool aGlobal )
177{
178 T tok;
179
180 while( ( tok = in->NextTok() ) != T_RIGHT && tok != T_EOF )
181 {
182 if( tok == T_LEFT )
183 tok = in->NextTok();
184
185 switch( tok )
186 {
187 case T_templatefields: // a token indicating class TEMPLATES.
188
189 // Be flexible regarding the starting point of the TEMPLATE_FIELDNAMES_LEXER
190 // stream. Caller may not have read the first two tokens out of the
191 // stream: T_LEFT and T_templatefields, so ignore them if seen here.
192 break;
193
194 case T_field:
195 {
196 // instantiate on stack, so if exception is thrown,
197 // destructor runs
198 TEMPLATE_FIELDNAME field;
199
200 field.Parse( in );
201
202 // add the field
203 if( !field.m_Name.IsEmpty() )
204 AddTemplateFieldName( field, aGlobal );
205 }
206 break;
207
208 default:
209 in->Unexpected( in->CurText() );
210 break;
211 }
212 }
213}
214
215
221{
223
224 // Note: order N^2 algorithm. Would need changing if fieldname template sets ever
225 // get large.
226
227 for( const TEMPLATE_FIELDNAME& global : m_globals )
228 {
229 bool overriddenInProject = false;
230
231 for( const TEMPLATE_FIELDNAME& project : m_project )
232 {
233 if( global.m_Name == project.m_Name )
234 {
235 overriddenInProject = true;
236 break;
237 }
238 }
239
240 if( !overriddenInProject )
241 m_resolved.push_back( global );
242 }
243
244 m_resolvedDirty = false;
245}
246
247
248void TEMPLATES::AddTemplateFieldName( const TEMPLATE_FIELDNAME& aFieldName, bool aGlobal )
249{
250 // Ensure that the template fieldname does not match a fixed fieldname.
251 for( FIELD_T fieldId : MANDATORY_FIELDS )
252 {
253 if( GetCanonicalFieldName( fieldId ) == aFieldName.m_Name )
254 return;
255 }
256
257 std::vector<TEMPLATE_FIELDNAME>& target = aGlobal ? m_globals : m_project;
258
259 // ensure uniqueness, overwrite any template fieldname by the same name.
260 for( TEMPLATE_FIELDNAME& temp : target )
261 {
262 if( temp.m_Name == aFieldName.m_Name )
263 {
264 temp = aFieldName;
265 m_resolvedDirty = true;
266 return;
267 }
268 }
269
270 // the name is legal and not previously added to the config container, append
271 // it and return its index within the container.
272 target.push_back( aFieldName );
273 m_resolvedDirty = true;
274}
275
276
277void TEMPLATES::AddTemplateFieldNames( const wxString& aSerializedFieldNames )
278{
279 TEMPLATE_FIELDNAMES_LEXER field_lexer( TO_UTF8( aSerializedFieldNames ) );
280
281 try
282 {
283 parse( &field_lexer, true );
284 }
285 catch( const IO_ERROR& )
286 {
287 }
288}
289
290
292{
293 if( aGlobal )
294 {
295 m_globals.clear();
297 }
298 else
299 {
300 m_project.clear();
302 }
303
304 m_resolvedDirty = false;
305}
306
307
308const std::vector<TEMPLATE_FIELDNAME>& TEMPLATES::GetTemplateFieldNames()
309{
310 if( m_resolvedDirty )
312
313 return m_resolved;
314}
315
316
317const std::vector<TEMPLATE_FIELDNAME>& TEMPLATES::GetTemplateFieldNames( bool aGlobal )
318{
319 if( aGlobal )
320 return m_globals;
321 else
322 return m_project;
323}
324
325
326const TEMPLATE_FIELDNAME* TEMPLATES::GetFieldName( const wxString& aName )
327{
328 if( m_resolvedDirty )
330
331 for( const TEMPLATE_FIELDNAME& field : m_resolved )
332 {
333 if( field.m_Name == aName )
334 return &field;
335 }
336
337 return nullptr;
338}
339
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
An interface used to output 8 bit text in a convenient way.
Definition: richio.h:322
std::string Quotew(const wxString &aWrapee) const
Definition: richio.cpp:545
int PRINTF_FUNC_N Print(int nestLevel, const char *fmt,...)
Format and write text to the output stream.
Definition: richio.cpp:460
void AddTemplateFieldName(const TEMPLATE_FIELDNAME &aFieldName, bool aGlobal)
Insert or append a wanted symbol field name into the field names template.
void AddTemplateFieldNames(const wxString &aSerializedFieldNames)
Add a serialized list of template field names.
const std::vector< TEMPLATE_FIELDNAME > & GetTemplateFieldNames()
Return a template field name list for read only access.
std::vector< TEMPLATE_FIELDNAME > m_resolved
void resolveTemplates()
Flatten project and global templates into a single list.
std::vector< TEMPLATE_FIELDNAME > m_project
const TEMPLATE_FIELDNAME * GetFieldName(const wxString &aName)
Search for aName in the template field name list.
void DeleteAllFieldNameTemplates(bool aGlobal)
Delete the entire contents.
void parse(TEMPLATE_FIELDNAMES_LEXER *in, bool aGlobal)
std::vector< TEMPLATE_FIELDNAME > m_globals
void Format(OUTPUTFORMATTER *out, bool aGlobal) const
Serialize this object out as text into the given OUTPUTFORMATTER.
#define _(s)
wxString From_UTF8(const char *cstring)
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
Definition: string_utils.h:403
Hold a name of a symbol's field, field value, and default visibility.
void Format(OUTPUTFORMATTER *out) const
Serialize this object out as text into the given OUTPUTFORMATTER.
void Parse(TEMPLATE_FIELDNAMES_LEXER *aSpec)
Fill this object from information in the input stream aSpec, which is a #TEMPLATE_FIELDNAMES_LEXER.
#define VALUE_CANONICAL
static wxString s_CanonicalSheetFile(SHEET_FILE_CANONICAL)
#define FOOTPRINT_CANONICAL
#define USER_FIELD_CANONICAL
#define INTERSHEET_REFS_CANONICAL
static wxString s_CanonicalReference(REFERENCE_CANONICAL)
static wxString s_CanonicalValue(VALUE_CANONICAL)
static wxString s_CanonicalFootprint(FOOTPRINT_CANONICAL)
static wxString s_CanonicalDatasheet(DATASHEET_CANONICAL)
#define DATASHEET_CANONICAL
wxString GetUserFieldName(int aFieldNdx, bool aTranslateForHI)
static wxString s_CanonicalIntersheetRefs(INTERSHEET_REFS_CANONICAL)
#define SHEET_NAME_CANONICAL
static wxString s_CanonicalSheetName(SHEET_NAME_CANONICAL)
#define DESCRIPTION_CANONICAL
static wxString s_CanonicalDescription(DESCRIPTION_CANONICAL)
#define REFERENCE_CANONICAL
#define SHEET_FILE_CANONICAL
wxString GetDefaultFieldName(FIELD_T aFieldId, bool aTranslateForHI)
Return a default symbol field name for a mandatory field type.
#define MANDATORY_FIELDS
FIELD_T
The set of all field indices assuming an array like sequence that a SCH_COMPONENT or LIB_PART can hol...
@ SHEET_FILENAME
@ INTERSHEET_REFS
Global label cross-reference page numbers.
@ DESCRIPTION
Field Description of part, i.e. "1/4W 1% Metal Film Resistor".
@ FOOTPRINT
Field Name Module PCB, i.e. "16DIP300".
@ DATASHEET
name of datasheet
@ REFERENCE
Field Reference of part, i.e. "IC21".
@ VALUE
Field Value of part, i.e. "3.3K".
wxString GetCanonicalFieldName(FIELD_T aFieldType)