summaryrefslogtreecommitdiff
path: root/pcbnew/dialogs/dialog_fp_plugin_options.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'pcbnew/dialogs/dialog_fp_plugin_options.cpp')
-rw-r--r--pcbnew/dialogs/dialog_fp_plugin_options.cpp399
1 files changed, 399 insertions, 0 deletions
diff --git a/pcbnew/dialogs/dialog_fp_plugin_options.cpp b/pcbnew/dialogs/dialog_fp_plugin_options.cpp
new file mode 100644
index 0000000..316ce5a
--- /dev/null
+++ b/pcbnew/dialogs/dialog_fp_plugin_options.cpp
@@ -0,0 +1,399 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2013 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
+ * Copyright (C) 2013 CERN
+ * Copyright (C) 2013 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
+ */
+
+
+#include <fctsys.h>
+#include <invoke_pcb_dialog.h>
+#include <dialog_fp_plugin_options_base.h>
+#include <fp_lib_table.h>
+#include <grid_tricks.h>
+
+
+#define INITIAL_HELP \
+ _( "Select an <b>Option Choice</b> in the listbox above, and then click the <b>Append Selected Option</b> button." )
+
+
+using std::string;
+
+// re-enter the dialog with the column sizes preserved from last time.
+static int col_width_option;
+static int col_width_value;
+
+
+/**
+ * Class DIALOG_FP_PLUGIN_OPTIONS
+ * is an options editor in the form of a two column name/value
+ * spreadsheet like (table) UI. It takes hints from a pcbnew PLUGIN as to
+ * supported options.
+ */
+class DIALOG_FP_PLUGIN_OPTIONS : public DIALOG_FP_PLUGIN_OPTIONS_BASE
+{
+
+public:
+ DIALOG_FP_PLUGIN_OPTIONS( wxTopLevelWindow* aParent,
+ const wxString& aNickname, const wxString& aPluginType,
+ const wxString& aOptions, wxString* aResult ) :
+ DIALOG_FP_PLUGIN_OPTIONS_BASE( aParent ),
+ m_callers_options( aOptions ),
+ m_result( aResult ),
+ m_initial_help( INITIAL_HELP )
+ {
+ wxString title = wxString::Format(
+ _( "Options for Library '%s'" ), GetChars( aNickname ) );
+
+ SetTitle( title );
+
+ // add Cut, Copy, and Paste to wxGrid
+ m_grid->PushEventHandler( new GRID_TRICKS( m_grid ) );
+
+ m_grid->SetColMinimalWidth( 1, 250 );
+
+ // Fill the grid with existing aOptions
+ string options = TO_UTF8( aOptions );
+
+ PROPERTIES* props = FP_LIB_TABLE::ParseOptions( options );
+
+ if( props )
+ {
+ if( (int) props->size() > m_grid->GetNumberRows() )
+ m_grid->AppendRows( props->size() - m_grid->GetNumberRows() );
+
+ int row = 0;
+ for( PROPERTIES::const_iterator it = props->begin(); it != props->end(); ++it, ++row )
+ {
+ m_grid->SetCellValue( row, 0, FROM_UTF8( it->first.c_str() ) );
+ m_grid->SetCellValue( row, 1, it->second );
+ }
+
+ delete props;
+ }
+
+ // Option Choices Panel:
+
+ IO_MGR::PCB_FILE_T pi_type = IO_MGR::EnumFromStr( aPluginType );
+ PLUGIN::RELEASER pi( IO_MGR::PluginFind( pi_type ) );
+
+ pi->FootprintLibOptions( &m_choices );
+
+ if( m_choices.size() )
+ {
+ int row = 0;
+ for( PROPERTIES::const_iterator it = m_choices.begin(); it != m_choices.end(); ++it, ++row )
+ {
+ wxString item = FROM_UTF8( it->first.c_str() );
+
+ m_listbox->InsertItems( 1, &item, row );
+ }
+ }
+
+ m_html->SetPage( m_initial_help );
+
+ if( !col_width_option )
+ {
+ m_grid->AutoSizeColumns( false );
+ }
+ else
+ {
+ m_grid->SetColSize( 0, col_width_option );
+ m_grid->SetColSize( 1, col_width_value );
+ }
+
+ Fit();
+
+ // initial focus on the grid please.
+ m_grid->SetFocus();
+ }
+
+ ~DIALOG_FP_PLUGIN_OPTIONS()
+ {
+ // destroy GRID_TRICKS before m_grid.
+ m_grid->PopEventHandler( true );
+ }
+
+
+private:
+ const wxString& m_callers_options;
+ wxString* m_result;
+ PROPERTIES m_choices;
+ wxString m_initial_help;
+
+
+ /// If the cursor is not on a valid cell, because there are no rows at all, return -1,
+ /// else return a 0 based column index.
+ int getCursorCol() const
+ {
+ return m_grid->GetGridCursorCol();
+ }
+
+ /// If the cursor is not on a valid cell, because there are no rows at all, return -1,
+ /// else return a 0 based row index.
+ int getCursorRow() const
+ {
+ return m_grid->GetGridCursorRow();
+ }
+
+ wxArrayString getRow( int aRow )
+ {
+ wxArrayString row;
+
+ const int col_count = m_grid->GetNumberCols();
+ for( int col = 0; col < col_count; ++col )
+ {
+ row.Add( m_grid->GetCellValue( aRow, col ) );
+ }
+
+ return row;
+ }
+
+ void setRow( int aRow, const wxArrayString& aPair )
+ {
+ const int col_count = m_grid->GetNumberCols();
+ for( int col = 0; col < col_count; ++col )
+ {
+ m_grid->SetCellValue( aRow, col, aPair[col] );
+ }
+ }
+
+ wxString makeResult()
+ {
+ PROPERTIES props;
+ const int rowCount = m_grid->GetNumberRows();
+
+ for( int row = 0; row<rowCount; ++row )
+ {
+ string name = TO_UTF8( m_grid->GetCellValue( row, 0 ).Trim( false ).Trim() );
+ UTF8 value = m_grid->GetCellValue( row, 1 ).Trim( false ).Trim();
+
+ if( name.size() )
+ {
+ props[name] = value;
+ }
+ }
+
+ return FP_LIB_TABLE::FormatOptions( &props );
+ }
+
+ void saveColSizes()
+ {
+ col_width_option = m_grid->GetColSize( 0 );
+ col_width_value = m_grid->GetColSize( 1 );
+ }
+
+ void abort()
+ {
+ saveColSizes();
+
+ *m_result = m_callers_options; // tell caller "no change"
+ EndModal( 0 );
+ }
+
+ int appendRow()
+ {
+ if( m_grid->AppendRows( 1 ) )
+ {
+ int last_row = m_grid->GetNumberRows() - 1;
+
+ // wx documentation is wrong, SetGridCursor does not make visible.
+ m_grid->MakeCellVisible( last_row, 0 );
+ m_grid->SetGridCursor( last_row, 0 );
+
+ return last_row;
+ }
+
+ return -1;
+ }
+
+ void appendOption()
+ {
+ int selected_row = m_listbox->GetSelection();
+ if( selected_row != wxNOT_FOUND )
+ {
+ wxString option = m_listbox->GetString( selected_row );
+
+ int row_count = m_grid->GetNumberRows();
+ int row;
+
+ for( row=0; row<row_count; ++row )
+ {
+ wxString col0 = m_grid->GetCellValue( row, 0 );
+
+ if( !col0 ) // empty col0
+ break;
+ }
+
+ if( row == row_count )
+ row = appendRow();
+
+ m_grid->SetCellValue( row, 0, option );
+ m_grid->AutoSizeColumns( false );
+ }
+ }
+
+ //-----<event handlers>------------------------------------------------------
+
+ void onListBoxItemSelected( wxCommandEvent& event )
+ {
+ // change the help text based on the m_listbox selection:
+ if( event.IsSelection() )
+ {
+ string option = TO_UTF8( event.GetString() );
+ UTF8 help_text;
+
+ if( m_choices.Value( option.c_str(), &help_text ) )
+ {
+ wxString page = help_text;
+
+ m_html->SetPage( page );
+ }
+ else
+ {
+ m_html->SetPage( m_initial_help );
+ }
+ }
+ }
+
+ void onListBoxItemDoubleClicked( wxCommandEvent& event )
+ {
+ appendOption();
+ }
+
+ void onAppendOption( wxCommandEvent& event )
+ {
+ appendOption();
+ }
+
+ void onAppendRow( wxMouseEvent& event )
+ {
+ appendRow();
+ }
+
+ void onDeleteRow( wxMouseEvent& event )
+ {
+ int rowCount = m_grid->GetNumberRows();
+ int curRow = getCursorRow();
+
+ m_grid->DeleteRows( curRow );
+
+ if( curRow && curRow == rowCount - 1 )
+ {
+ m_grid->MakeCellVisible( curRow-1, getCursorCol() );
+ m_grid->SetGridCursor( curRow-1, getCursorCol() );
+ }
+ }
+
+ void onMoveUp( wxMouseEvent& event )
+ {
+ int curRow = getCursorRow();
+ if( curRow >= 1 )
+ {
+ int curCol = getCursorCol();
+
+ wxArrayString move_me = getRow( curRow );
+
+ m_grid->DeleteRows( curRow );
+ --curRow;
+ m_grid->InsertRows( curRow );
+
+ setRow( curRow, move_me );
+
+ wxGridTableBase* tbl = m_grid->GetTable();
+
+ if( tbl->GetView() )
+ {
+ // fire a msg to cause redrawing
+ wxGridTableMessage msg( tbl,
+ wxGRIDTABLE_NOTIFY_ROWS_INSERTED,
+ curRow,
+ 0 );
+
+ tbl->GetView()->ProcessTableMessage( msg );
+ }
+
+ m_grid->MakeCellVisible( curRow, curCol );
+ m_grid->SetGridCursor( curRow, curCol );
+ }
+ }
+
+ void onMoveDown( wxMouseEvent& event )
+ {
+ int curRow = getCursorRow();
+ if( curRow + 1 < m_grid->GetNumberRows() )
+ {
+ int curCol = getCursorCol();
+
+ wxArrayString move_me = getRow( curRow );
+
+ m_grid->DeleteRows( curRow );
+ ++curRow;
+ m_grid->InsertRows( curRow );
+ setRow( curRow, move_me );
+
+ wxGridTableBase* tbl = m_grid->GetTable();
+
+ if( tbl->GetView() )
+ {
+ // fire a msg to cause redrawing
+ wxGridTableMessage msg( tbl,
+ wxGRIDTABLE_NOTIFY_ROWS_INSERTED,
+ curRow - 1,
+ 0 );
+
+ tbl->GetView()->ProcessTableMessage( msg );
+ }
+
+ m_grid->MakeCellVisible( curRow, curCol );
+ m_grid->SetGridCursor( curRow, curCol );
+ }
+ }
+
+ void onCancelButtonClick( wxCommandEvent& event )
+ {
+ abort();
+ }
+
+ void onCancelCaptionButtonClick( wxCloseEvent& event )
+ {
+ abort();
+ }
+
+ void onOKButtonClick( wxCommandEvent& event )
+ {
+ saveColSizes();
+
+ *m_result = makeResult(); // change from edits
+ EndModal( 1 );
+ }
+ //-----</event handlers>-----------------------------------------------------
+};
+
+
+void InvokePluginOptionsEditor( wxTopLevelWindow* aCaller,
+ const wxString& aNickname, const wxString& aPluginType,
+ const wxString& aOptions, wxString* aResult )
+{
+ DIALOG_FP_PLUGIN_OPTIONS dlg( aCaller, aNickname, aPluginType, aOptions, aResult );
+
+ dlg.ShowModal();
+}