/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2013 Jean-Pierre Charras, jp.charras@wanadoo.fr * Copyright (C) 1992-2013 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 */ /** * @file eeschema/dialogs/dialog_bom.cpp * @brief Dialog box for creating bom and other documents from generic netlist. */ #include #include #include #include #include #include #include #include #include #include #include #define BOM_PLUGINS_KEY wxT("bom_plugins") #define BOM_PLUGIN_SELECTED_KEY wxT("bom_plugin_selected") const char * s_bomHelpInfo = #include ; #include using namespace T_BOMCFG_T; /** * Class BOM_CFG_READER_PARSER * holds data and functions pertinent to parsing a S-expression file * for a WORKSHEET_LAYOUT. */ class BOM_CFG_READER_PARSER : public DIALOG_BOM_CFG_LEXER { wxArrayString* m_pluginsList; public: BOM_CFG_READER_PARSER( wxArrayString* aPlugins, const char* aData, const wxString& aSource ); void Parse() throw( PARSE_ERROR, IO_ERROR ); private: void parsePlugin() throw( IO_ERROR, PARSE_ERROR ); }; // PCB_PLOT_PARAMS_PARSER BOM_CFG_READER_PARSER::BOM_CFG_READER_PARSER( wxArrayString* aPlugins, const char* aLine, const wxString& aSource ) : DIALOG_BOM_CFG_LEXER( aLine, aSource ) { m_pluginsList = aPlugins; } void BOM_CFG_READER_PARSER::Parse() throw( PARSE_ERROR, IO_ERROR ) { T token; while( ( token = NextTok() ) != T_RIGHT ) { if( token == T_EOF) break; if( token == T_LEFT ) token = NextTok(); if( token == T_plugins ) continue; switch( token ) { case T_plugin: // Defines a new plugin parsePlugin(); break; default: // Unexpected( CurText() ); break; } } } void BOM_CFG_READER_PARSER::parsePlugin() throw( IO_ERROR, PARSE_ERROR ) { wxString title, command; NeedSYMBOLorNUMBER(); title = FromUTF8(); T token; while( ( token = NextTok() ) != T_RIGHT ) { if( token == T_EOF) break; switch( token ) { case T_LEFT: break; case T_cmd: NeedSYMBOLorNUMBER(); command = FromUTF8(); NeedRIGHT(); break; case T_opts: while( ( token = NextTok() ) != T_RIGHT && token != T_EOF ); break; default: Unexpected( CurText() ); break; } } if( ! title.IsEmpty() ) { m_pluginsList->Add( title ); m_pluginsList->Add( command ); } } // The main dialog frame tu run scripts to build bom class DIALOG_BOM : public DIALOG_BOM_BASE { private: SCH_EDIT_FRAME* m_parent; // The list of scripts (or plugins): // a script descr uses 2 lines: // the first is the title // the second is the command line wxArrayString m_plugins; wxConfig* m_config; // to store the "plugins" public: // Constructor and destructor DIALOG_BOM( SCH_EDIT_FRAME* parent ); ~DIALOG_BOM(); private: void OnPluginSelected( wxCommandEvent& event ); void OnRunPlugin( wxCommandEvent& event ); void OnCancelClick( wxCommandEvent& event ); void OnHelp( wxCommandEvent& event ); void OnAddPlugin( wxCommandEvent& event ); void OnChoosePlugin( wxCommandEvent& event ); void OnRemovePlugin( wxCommandEvent& event ); void OnEditPlugin( wxCommandEvent& event ); void OnCommandLineEdited( wxCommandEvent& event ); void OnNameEdited( wxCommandEvent& event ); void pluginInit(); void installPluginsList(); }; // Create and show DIALOG_BOM. int InvokeDialogCreateBOM( SCH_EDIT_FRAME* aCaller ) { DIALOG_BOM dlg( aCaller ); return dlg.ShowModal(); } DIALOG_BOM::DIALOG_BOM( SCH_EDIT_FRAME* parent ) : DIALOG_BOM_BASE( parent ) { m_parent = parent; m_config = wxGetApp().GetSettings(); installPluginsList(); GetSizer()->SetSizeHints( this ); Centre(); } DIALOG_BOM::~DIALOG_BOM() { // Save the plugin descriptions in config. // the config stores only one string. // plugins are saved inside a S expr: // ( plugins // ( plugin "plugin name" (cmd "command line") ) // .... // ) STRING_FORMATTER writer; writer.Print( 0, "(plugins" ); for( unsigned ii = 0; ii < m_plugins.GetCount(); ii += 2 ) { writer.Print( 1, "(plugin %s (cmd %s))", writer.Quotew( m_plugins[ii] ).c_str(), writer.Quotew( m_plugins[ii+1] ).c_str() ); } writer.Print( 0, ")" ); wxString list( FROM_UTF8( writer.GetString().c_str() ) ); m_config->Write( BOM_PLUGINS_KEY, list ); wxString active_plugin_name = m_lbPlugins->GetStringSelection( ); m_config->Write( BOM_PLUGIN_SELECTED_KEY, active_plugin_name ); } /* Read the initialized plugins in config and fill the list * of names */ void DIALOG_BOM::installPluginsList() { wxString list, active_plugin_name; m_config->Read( BOM_PLUGINS_KEY, &list ); m_config->Read( BOM_PLUGIN_SELECTED_KEY, &active_plugin_name ); if( !list.IsEmpty() ) { BOM_CFG_READER_PARSER cfg_parser( &m_plugins, TO_UTF8( list ), wxT( "plugins" ) ); try { cfg_parser.Parse(); } catch( IO_ERROR ioe ) { // wxLogMessage( ioe.errorText ); } } // Populate list box for( unsigned ii = 0; ii < m_plugins.GetCount(); ii+=2 ) { m_lbPlugins->Append( m_plugins[ii] ); if( active_plugin_name == m_plugins[ii] ) m_lbPlugins->SetSelection( ii/2 ); } pluginInit(); } void DIALOG_BOM::OnPluginSelected( wxCommandEvent& event ) { pluginInit(); } void DIALOG_BOM::pluginInit() { int ii = m_lbPlugins->GetSelection(); if( ii < 0 ) { m_textCtrlName->SetValue( wxEmptyString ); m_textCtrlCommand->SetValue( wxEmptyString ); return; } m_textCtrlName->SetValue( m_plugins[2 * ii] ); m_textCtrlCommand->SetValue( m_plugins[(2 * ii)+1] ); } /** * Function RunPlugin * run the plugin command line */ void DIALOG_BOM::OnRunPlugin( wxCommandEvent& event ) { wxFileName fn; wxString fileWildcard; wxString title = _( "Save Netlist File" ); // Calculate the xml netlist filename fn = g_RootSheet->GetScreen()->GetFileName(); if( fn.GetPath().IsEmpty() ) fn.SetPath( wxGetCwd() ); fn.ClearExt(); wxString fullfilename = fn.GetFullPath(); m_parent->ClearMsgPanel(); m_parent->SetNetListerCommand( m_textCtrlCommand->GetValue() ); m_parent->CreateNetlist( -1, fullfilename, 0 ); } void DIALOG_BOM::OnCancelClick( wxCommandEvent& event ) { EndModal( wxID_CANCEL ); } /** * Function OnRemovePlugin * Remove a plugin from the list */ void DIALOG_BOM::OnRemovePlugin( wxCommandEvent& event ) { int ii = m_lbPlugins->GetSelection(); if( ii < 0 ) return; m_lbPlugins->Delete( ii ); m_plugins.RemoveAt( 2*ii, 2 ); // Remove title and command line // Select the next item, if exists if( (int)m_lbPlugins->GetCount() >= ii ) ii = m_lbPlugins->GetCount() - 1; if( ii >= 0 ) m_lbPlugins->SetSelection( ii ); pluginInit(); } /** * Function OnAddPlugin * Add a new panel for a new netlist plugin */ void DIALOG_BOM::OnAddPlugin( wxCommandEvent& event ) { // Creates a new plugin entry wxString name = wxGetTextFromUser( _("Plugin") ); if( name.IsEmpty() ) return; // Verify if it does not exists for( unsigned ii = 0; ii < m_plugins.GetCount(); ii += 2 ) { if( name == m_plugins[ii] ) { wxMessageBox( _("This plugin already exists. Abort") ); return; } } m_plugins.Add( name ); m_plugins.Add( wxEmptyString ); m_lbPlugins->Append( name ); m_lbPlugins->SetSelection( m_lbPlugins->GetCount() - 1 ); pluginInit(); } /* * Browse plugin files, and set m_CommandStringCtrl field */ void DIALOG_BOM::OnChoosePlugin( wxCommandEvent& event ) { wxString FullFileName, Mask, Path; Mask = wxT( "*" ); Path = wxGetApp().GetExecutablePath(); FullFileName = EDA_FileSelector( _( "Plugin files:" ), Path, FullFileName, wxEmptyString, Mask, this, wxFD_OPEN, true ); if( FullFileName.IsEmpty() ) return; // Creates a default command line, // suitable to run the external tool xslproc or python // The default command line depending on plugin extension, currently // "xsl" or "exe" or "py" wxString cmdLine; wxFileName fn( FullFileName ); wxString ext = fn.GetExt(); if( ext == wxT("xsl" ) ) cmdLine.Printf(wxT("xsltproc -o \"%%O\" \"%s\" \"%%I\""), GetChars(FullFileName) ); else if( ext == wxT("exe" ) || ext.IsEmpty() ) cmdLine.Printf(wxT("\"%s\" < \"%%I\" > \"%%O\""), GetChars(FullFileName) ); else if( ext == wxT("py" ) || ext.IsEmpty() ) cmdLine.Printf(wxT("python \"%s\" \"%%I\" \"%%O\""), GetChars(FullFileName) ); else cmdLine.Printf(wxT("\"%s\""), GetChars(FullFileName) ); m_textCtrlCommand->SetValue( cmdLine ); } void DIALOG_BOM::OnEditPlugin( wxCommandEvent& event ) { wxString pluginName, cmdline; // Try to find the plugin name. // This is possible if the name ends by .py or .xsl cmdline = m_textCtrlCommand->GetValue(); int pos = -1; if( (pos = cmdline.Find( wxT(".py") )) != wxNOT_FOUND ) pos += 2; else if( (pos = cmdline.Find( wxT(".xsl") )) != wxNOT_FOUND ) pos += 3; // the end of plugin name is at position pos. if( pos > 0 ) { // Be sure this is the end of the name: the next char is " or space int eos = cmdline[pos+1]; if( eos == ' '|| eos == '\"' ) { // search for the starting point of the name int jj = pos-1; while( jj >= 0 ) if( cmdline[jj] != eos ) jj--; else break; // extract the name if( jj >= 0 ) pluginName = cmdline.SubString( jj, pos ); } } AddDelimiterString( pluginName ); wxString editorname = wxGetApp().GetEditorName(); if( !editorname.IsEmpty() ) ExecuteFile( this, editorname, pluginName ); else wxMessageBox( _("No text editor selected in KiCad. Please choose it") ); } void DIALOG_BOM::OnHelp( wxCommandEvent& event ) { HTML_MESSAGE_BOX help_Dlg( this, _("Bom Generation Help"), wxDefaultPosition, wxSize( 750,550 ) ); wxString msg = FROM_UTF8(s_bomHelpInfo); help_Dlg.m_htmlWindow->AppendToPage( msg ); help_Dlg.ShowModal(); } void DIALOG_BOM::OnCommandLineEdited( wxCommandEvent& event ) { int ii = m_lbPlugins->GetSelection(); if( ii < 0 ) return; m_plugins[(2 * ii)+1] = m_textCtrlCommand->GetValue(); } void DIALOG_BOM::OnNameEdited( wxCommandEvent& event ) { int ii = m_lbPlugins->GetSelection(); if( ii < 0 ) return; m_plugins[2 * ii] = m_textCtrlName->GetValue(); m_lbPlugins->SetString( ii, m_plugins[2 * ii] ); }