KiCad PCB EDA Suite
Loading...
Searching...
No Matches
pcb_edit_table_tool.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 The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <https://www.gnu.org/licenses/>.
18 */
19
20#include <tool/tool_manager.h>
22#include <tools/pcb_actions.h>
23#include <collectors.h>
24#include <dialogs/dialog_table_properties.h>
26#include <wx/filedlg.h>
27#include <wx/msgdlg.h>
28#include <fstream>
29#include <kiplatform/ui.h>
30
31
33 PCB_TOOL_BASE( "pcbnew.TableEditor" )
34{
35}
36
37
39{
41
43
44 return true;
45}
46
47
49{
50 // Use copy constructor to copy all formatting properties (font, colors, borders, etc.)
51 PCB_TABLECELL* cell = new PCB_TABLECELL( *aSource );
52
53 // Generate a new UUID to avoid duplicates (copy constructor preserves the old UUID)
54 cell->ResetUuidDirect();
55
56 // Clear text content - we only want the formatting, not the content
57 cell->SetText( wxEmptyString );
58
59 // Position will be set by the caller, but preserve size from source
60 cell->SetStart( aSource->GetStart() );
61 cell->SetEnd( aSource->GetEnd() );
62
63 return cell;
64}
65
66
68{
69 PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
70
71 return selTool->RequestSelection(
72 []( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector, PCB_SELECTION_TOOL* sTool )
73 {
74 // Iterate from the back so we don't have to worry about removals.
75 for( int i = aCollector.GetCount() - 1; i >= 0; --i )
76 {
77 if( !dynamic_cast<PCB_TABLECELL*>( aCollector[i] ) )
78 aCollector.Remove( aCollector[i] );
79 }
80 } );
81}
82
83
88
89
91{
93 bool clearSelection = selection.IsHover();
94 PCB_TABLE* parentTable = nullptr;
95
96 for( EDA_ITEM* item : selection.Items() )
97 {
98 if( item->Type() != PCB_TABLECELL_T )
99 return 0;
100
101 PCB_TABLE* table = static_cast<PCB_TABLE*>( item->GetParent() );
102
103 if( !parentTable )
104 {
105 parentTable = table;
106 }
107 else if( parentTable != table )
108 {
109 parentTable = nullptr;
110 break;
111 }
112 }
113
114 if( parentTable )
115 {
116 DIALOG_TABLE_PROPERTIES dlg( frame(), parentTable );
117
118 dlg.ShowQuasiModal(); // Scintilla's auto-complete requires quasiModal
119 }
120
121 if( clearSelection )
123
124 return 0;
125}
126
127
129{
131 bool clearSelection = selection.IsHover();
132 PCB_TABLE* parentTable = nullptr;
133
134 // Find the table from the selection
135 for( EDA_ITEM* item : selection.Items() )
136 {
137 if( item->Type() != PCB_TABLECELL_T )
138 return 0;
139
140 PCB_TABLE* table = static_cast<PCB_TABLE*>( item->GetParent() );
141
142 if( !parentTable )
143 {
144 parentTable = table;
145 }
146 else if( parentTable != table )
147 {
148 parentTable = nullptr;
149 break;
150 }
151 }
152
153 if( !parentTable )
154 return 0;
155
156 // Show file save dialog
157 wxFileDialog saveDialog( frame(), _( "Export Table to CSV" ),
158 wxEmptyString, wxEmptyString,
159 _( "CSV files (*.csv)|*.csv" ),
160 wxFD_SAVE | wxFD_OVERWRITE_PROMPT );
161
163
164 if( saveDialog.ShowModal() == wxID_CANCEL )
165 {
166 if( clearSelection )
168 return 0;
169 }
170
171 wxString filePath = saveDialog.GetPath();
172
173 // Ensure .csv extension
174 if( !filePath.Lower().EndsWith( ".csv" ) )
175 filePath += ".csv";
176
177 // Open file for writing
178 std::ofstream outFile( filePath.ToStdString() );
179
180 if( !outFile.is_open() )
181 {
182 wxMessageBox( wxString::Format( _( "Failed to open file:\n%s" ), filePath ),
183 _( "Export Error" ), wxOK | wxICON_ERROR, frame() );
184
185 if( clearSelection )
187 return 0;
188 }
189
190 // Helper function to escape CSV fields
191 auto escapeCSV = []( const wxString& field ) -> wxString
192 {
193 wxString escaped = field;
194
195 // If field contains comma, quote, or newline, wrap in quotes and escape quotes
196 if( escaped.Contains( ',' ) || escaped.Contains( '\"' ) || escaped.Contains( '\n' ) )
197 {
198 escaped.Replace( "\"", "\"\"" ); // Escape quotes by doubling them
199 escaped = "\"" + escaped + "\"";
200 }
201
202 return escaped;
203 };
204
205 // Export table data
206 for( int row = 0; row < parentTable->GetRowCount(); ++row )
207 {
208 for( int col = 0; col < parentTable->GetColCount(); ++col )
209 {
210 PCB_TABLECELL* cell = parentTable->GetCell( row, col );
211
212 // Get resolved text (with variables expanded)
213 wxString cellText = cell->GetShownText( false, 0 );
214
215 // Write escaped cell text
216 outFile << escapeCSV( cellText ).ToStdString();
217
218 // Add comma separator unless it's the last column
219 if( col < parentTable->GetColCount() - 1 )
220 outFile << ',';
221 }
222
223 // End of row
224 outFile << '\n';
225 }
226
227 outFile.close();
228
229 if( clearSelection )
231
232 return 0;
233}
234
235
static TOOL_ACTION addRowAbove
Definition actions.h:100
static TOOL_ACTION addColBefore
Definition actions.h:102
static TOOL_ACTION deleteRows
Definition actions.h:104
static TOOL_ACTION addRowBelow
Definition actions.h:101
static TOOL_ACTION deleteColumns
Definition actions.h:105
static TOOL_ACTION unmergeCells
Definition actions.h:107
static TOOL_ACTION mergeCells
Definition actions.h:106
static TOOL_ACTION addColAfter
Definition actions.h:103
static TOOL_ACTION editTable
Definition actions.h:108
static TOOL_ACTION selectionClear
Clear the current selection.
Definition actions.h:220
static TOOL_ACTION exportTableCSV
Definition actions.h:109
void ResetUuidDirect()
Definition board_item.h:242
int GetCount() const
Return the number of objects in the list.
Definition collector.h:79
void Remove(int aIndex)
Remove the item at aIndex (first position is 0).
Definition collector.h:107
A base class for most all the KiCad significant classes used in schematics and boards.
Definition eda_item.h:96
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition eda_shape.h:240
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition eda_shape.h:190
virtual void SetText(const wxString &aText)
Definition eda_text.cpp:265
Used when the right click button is pressed, or when the select tool is in effect.
Definition collectors.h:203
int AddRowAbove(const TOOL_EVENT &aEvent)
int EditTable(const TOOL_EVENT &aEvent)
int DeleteRows(const TOOL_EVENT &aEvent)
int AddRowBelow(const TOOL_EVENT &aEvent)
bool Init() override
Init() is called once upon a registration of the tool.
void setTransitions() override
< Set up handlers for various events.
int UnmergeCells(const TOOL_EVENT &aEvent)
int AddColumnAfter(const TOOL_EVENT &aEvent)
int DeleteColumns(const TOOL_EVENT &aEvent)
int AddColumnBefore(const TOOL_EVENT &aEvent)
void clearSelection() override
const SELECTION & getTableCellSelection() override
int MergeCells(const TOOL_EVENT &aEvent)
int ExportTableToCSV(const TOOL_EVENT &aEvent)
PCB_TABLECELL * copyCell(PCB_TABLECELL *aSource) override
The selection tool: currently supports:
PCB_SELECTION & RequestSelection(CLIENT_SELECTION_FILTER aClientFilter)
Return the current selection, filtered according to aClientFilter.
void SetEnd(const VECTOR2I &aEnd) override
void SetStart(const VECTOR2I &aStart) override
wxString GetShownText(bool aAllowExtraText, int aDepth=0) const override
Return the string actually shown after processing of the base text.
int GetRowCount() const
Definition pcb_table.h:121
PCB_TABLECELL * GetCell(int aRow, int aCol) const
Definition pcb_table.h:146
int GetColCount() const
Definition pcb_table.h:119
T * frame() const
PCB_TOOL_BASE(TOOL_ID aId, const std::string &aName)
Constructor.
virtual bool Init() override
Init() is called once upon a registration of the tool.
const PCB_SELECTION & selection() const
TOOL_MANAGER * m_toolMgr
Definition tool_base.h:220
Generic, UI-independent tool event.
Definition tool_event.h:167
void Go(int(T::*aStateFunc)(const TOOL_EVENT &), const TOOL_EVENT_LIST &aConditions=TOOL_EVENT(TC_ANY, TA_ANY))
Define which state (aStateFunc) to go when a certain event arrives (aConditions).
TOOL_MENU & GetToolMenu()
CONDITIONAL_MENU & GetMenu()
Definition tool_menu.cpp:40
#define _(s)
void AllowNetworkFileSystems(wxDialog *aDialog)
Configure a file dialog to show network and virtual file systems.
Definition wxgtk/ui.cpp:448
std::vector< std::vector< std::string > > table
@ PCB_TABLECELL_T
class PCB_TABLECELL, PCB_TEXTBOX for use in tables
Definition typeinfo.h:88
VECTOR2< int32_t > VECTOR2I
Definition vector2d.h:683