29using namespace std::placeholders;
104 int preExisting = commandToUndo->
GetCount();
106 for(
unsigned ii = 0; ii < aItemsList.
GetCount(); ii++ )
109 for(
unsigned ii = preExisting; ii < commandToUndo->
GetCount(); ii++ )
114 if( command == UNDO_REDO::UNSPECIFIED )
116 command = aCommandType;
124 case UNDO_REDO::CHANGED:
125 case UNDO_REDO::DRILLORIGIN:
126 case UNDO_REDO::GRIDORIGIN:
133 case UNDO_REDO::NEWITEM:
134 case UNDO_REDO::DELETED:
135 case UNDO_REDO::PAGESETTINGS:
136 case UNDO_REDO::REGROUP:
137 case UNDO_REDO::UNGROUP:
141 wxFAIL_MSG( wxString::Format( wxT(
"Unrecognized undo command: %X" ), command ) );
158 delete commandToUndo;
260 bool not_found =
false;
261 bool reBuild_ratsnest =
false;
262 bool deep_reBuild_ratsnest =
false;
263 bool solder_mask_dirty =
false;
272 enum ITEM_CHANGE_TYPE
279 std::unordered_map<EDA_ITEM*, ITEM_CHANGE_TYPE> item_changes;
281 auto update_item_change_state =
282 [&](
EDA_ITEM* item, ITEM_CHANGE_TYPE change_type )
284 auto item_itr = item_changes.find( item );
286 if( item_itr == item_changes.end() )
289 item_changes.insert( { item, change_type } );
294 switch( item_itr->second )
296 case ITEM_CHANGE_TYPE::ADDED:
298 if( change_type == ITEM_CHANGE_TYPE::DELETED )
302 item_changes.erase( item_itr );
304 else if( change_type == ITEM_CHANGE_TYPE::ADDED )
307 wxASSERT_MSG(
false, wxT(
"UndoRedo: should not add already added item" ) );
313 case ITEM_CHANGE_TYPE::DELETED:
317 wxASSERT_MSG(
false, wxT(
"UndoRedo: should not alter already deleted item" ) );
320 case ITEM_CHANGE_TYPE::CHANGED:
322 if( change_type == ITEM_CHANGE_TYPE::DELETED )
324 item_itr->second = ITEM_CHANGE_TYPE::DELETED;
326 else if( change_type == ITEM_CHANGE_TYPE::ADDED )
331 wxT(
"UndoRedo: should not add already changed item" ) );
344 for(
int ii = (
int) aList->
GetCount() - 1; ii >= 0 ; ii-- )
357 if( status != UNDO_REDO::DELETED
358 && status != UNDO_REDO::REGROUP
359 && status != UNDO_REDO::UNGROUP
360 && status != UNDO_REDO::DRILLORIGIN
361 && status != UNDO_REDO::GRIDORIGIN
362 && status != UNDO_REDO::PAGESETTINGS )
367 wxASSERT_MSG(
false, wxT(
"Item in the undo buffer does not exist" ) );
381 switch( eda_item->
Type() )
384 deep_reBuild_ratsnest =
true;
392 reBuild_ratsnest =
true;
396 reBuild_ratsnest =
true;
397 deep_reBuild_ratsnest =
true;
404 switch( eda_item->
Type() )
407 solder_mask_dirty =
true;
411 solder_mask_dirty =
true;
420 LSET layers =
static_cast<BOARD_ITEM*
>( eda_item )->GetLayerSet();
423 solder_mask_dirty =
true;
434 case UNDO_REDO::CHANGED:
450 view->Remove( item );
471 view->Hide( item,
false );
477 update_item_change_state( item, ITEM_CHANGE_TYPE::CHANGED );
481 case UNDO_REDO::NEWITEM:
484 update_item_change_state( eda_item, ITEM_CHANGE_TYPE::DELETED );
487 view->Remove( eda_item );
493 case UNDO_REDO::DELETED:
499 update_item_change_state( eda_item, ITEM_CHANGE_TYPE::ADDED );
502 view->Add( eda_item );
506 case UNDO_REDO::REGROUP:
517 group->RemoveItem( boardItem );
523 case UNDO_REDO::UNGROUP:
534 group->AddItem( boardItem );
539 case UNDO_REDO::DRILLORIGIN:
540 case UNDO_REDO::GRIDORIGIN:
556 case UNDO_REDO::PAGESETTINGS:
562 *item = std::move( alt_item );
567 wxFAIL_MSG( wxString::Format( wxT(
"PutDataInPreviousState() error (unknown code %X)" ),
574 wxMessageBox(
_(
"Incomplete undo/redo operation: some items not found" ) );
578 if( reBuild_ratsnest || deep_reBuild_ratsnest )
585 if( solder_mask_dirty )
595 std::vector<BOARD_ITEM*> added_items, deleted_items, changed_items;
597 for(
auto& [item, changeType] : item_changes )
601 case ITEM_CHANGE_TYPE::ADDED:
602 added_items.push_back(
static_cast<BOARD_ITEM*
>( item ) );
605 case ITEM_CHANGE_TYPE::DELETED:
606 deleted_items.push_back(
static_cast<BOARD_ITEM*
>( item ) );
609 case ITEM_CHANGE_TYPE::CHANGED:
610 changed_items.push_back(
static_cast<BOARD_ITEM*
>( item ) );
615 if( added_items.size() > 0 || deleted_items.size() > 0 || changed_items.size() > 0 )
622 if( aItemCount == 0 )
633 for(
int ii = 0; ii < aItemCount; ii++ )
653 "Item on undo/redo list not owned by undo/redo!" );
656 static_cast<BOARD_ITEM*
>( item )->SetParentGroup(
nullptr );
static EDA_ITEM * MakeImage(EDA_ITEM *aItem)
static void DoSetDrillOrigin(KIGFX::VIEW *aView, PCB_BASE_FRAME *aFrame, EDA_ITEM *aItem, const VECTOR2D &aPoint)
Abstract interface for BOARD_ITEMs capable of storing other items inside.
virtual void Remove(BOARD_ITEM *aItem, REMOVE_MODE aMode=REMOVE_MODE::NORMAL)=0
Removes an item from the container.
virtual void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false)=0
Adds an item to the container.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
void SetParentGroup(PCB_GROUP *aGroup)
void SwapItemData(BOARD_ITEM *aImage)
Swap data between aItem and aImage.
PCB_GROUP * GetParentGroup() const
FOOTPRINT * GetParentFootprint() const
BOARD_ITEM * GetItem(const KIID &aID) const
bool BuildConnectivity(PROGRESS_REPORTER *aReporter=nullptr)
Build or rebuild the board connectivity database for the board, especially the list of connected item...
void OnItemsCompositeUpdate(std::vector< BOARD_ITEM * > &aAddedItems, std::vector< BOARD_ITEM * > &aRemovedItems, std::vector< BOARD_ITEM * > &aChangedItems)
Notify the board and its listeners that items on the board have been modified in a composite operatio...
void IncrementTimeStamp()
std::shared_ptr< CONNECTIVITY_DATA > GetConnectivity() const
Return a list of missing connections between components/tracks.
static DELETED_BOARD_ITEM * GetInstance()
void Restore(EDA_DRAW_FRAME *aFrame, KIGFX::VIEW *aView=nullptr)
virtual void PushCommandToUndoList(PICKED_ITEMS_LIST *aItem)
Add a command to undo in the undo list.
virtual int GetRedoCommandCount() const
UNDO_REDO_CONTAINER m_undoList
UNDO_REDO_LIST
Specify whether we are interacting with the undo or redo stacks.
virtual PICKED_ITEMS_LIST * PopCommandFromRedoList()
Return the last command to undo and remove it from list, nothing is deleted.
UNDO_REDO_CONTAINER m_redoList
virtual PICKED_ITEMS_LIST * PopCommandFromUndoList()
Return the last command to undo and remove it from list, nothing is deleted.
virtual int GetUndoCommandCount() const
virtual void PushCommandToRedoList(PICKED_ITEMS_LIST *aItem)
Add a command to redo in the redo list.
bool IsType(FRAME_T aType) const
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
A base class for most all the KiCad significant classes used in schematics and boards.
virtual VECTOR2I GetPosition() const
void SetFlags(EDA_ITEM_FLAGS aMask)
KICAD_T Type() const
Returns the type of object.
void ClearFlags(EDA_ITEM_FLAGS aMask=EDA_ITEM_ALL_FLAGS)
bool HasFlag(EDA_ITEM_FLAGS aFlag) const
static const TOOL_EVENT UndoRedoPreEvent
static const TOOL_EVENT SelectedItemsModified
Selected items were moved, this can be very high frequency on the canvas, use with care.
static const TOOL_EVENT UndoRedoPostEvent
bool IsBOARD_ITEM() const
LSET is a set of PCB_LAYER_IDs.
void ClearUndoORRedoList(UNDO_REDO_LIST whichList, int aItemCount=-1) override
Free the undo or redo list from List element.
void RestoreCopyFromUndoList(wxCommandEvent &aEvent)
Undo the last edit:
void saveCopyInUndoList(PICKED_ITEMS_LIST *commandToUndo, const PICKED_ITEMS_LIST &aItemsList, UNDO_REDO aCommandType)
void AppendCopyToUndoList(const PICKED_ITEMS_LIST &aItemsList, UNDO_REDO aCommandType) override
As SaveCopyInUndoList, but appends the changes to the last undo item on the stack.
void SaveCopyInUndoList(EDA_ITEM *aItemToCopy, UNDO_REDO aTypeCommand) override
Create a new entry in undo list of commands.
void ClearListAndDeleteItems(PICKED_ITEMS_LIST *aList)
void RollbackFromUndo()
Perform an undo of the last edit without logging a corresponding redo.
bool UndoRedoBlocked() const
Check if the undo and redo operations are currently blocked.
void PutDataInPreviousState(PICKED_ITEMS_LIST *aList)
Used in undo or redo command.
void RestoreCopyFromRedoList(wxCommandEvent &aEvent)
Redo the last edit:
EDA_ITEM * GetItem(const KIID &aId) const override
Fetch an item by KIID.
void OnModify() override
Must be called after a change in order to set the "modify" flag and update other data structures and ...
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual BOARD_ITEM_CONTAINER * GetModel() const =0
void Compile_Ratsnest(bool aDisplayStatus)
Create the entire board ratsnest.
static void DoSetGridOrigin(KIGFX::VIEW *aView, PCB_BASE_FRAME *aFrame, EDA_ITEM *originViewItem, const VECTOR2D &aPoint)
virtual KIGFX::PCB_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
A set of BOARD_ITEMs (i.e., without duplicates).
virtual bool RemoveItem(BOARD_ITEM *aItem)
Remove item from group.
virtual bool AddItem(BOARD_ITEM *aItem)
Add item to group.
A holder to handle information on schematic or board items.
bool SetPickedItemStatus(UNDO_REDO aStatus, unsigned aIdx)
Set the type of undo/redo operation for a given picked item.
void PushItem(const ITEM_PICKER &aItem)
Push aItem to the top of the list.
void SetDescription(const wxString &aDescription)
UNDO_REDO GetPickedItemStatus(unsigned int aIdx) const
ITEM_PICKER GetItemWrapper(unsigned int aIdx) const
EDA_ITEM * GetPickedItemLink(unsigned int aIdx) const
wxString GetDescription() const
bool RemovePicker(unsigned aIdx)
Remove one entry (one picker) from the list of picked items.
KIID GetPickedItemGroupId(unsigned int aIdx) const
unsigned GetCount() const
void ReversePickersListOrder()
Reverse the order of pickers stored in this list.
bool SetPickedItemLink(EDA_ITEM *aLink, unsigned aIdx)
Set the link associated to a given picked item.
void ClearListAndDeleteItems(std::function< void(EDA_ITEM *)> aItemDeleter)
Delete the list of pickers AND the data pointed by #m_PickedItem or #m_PickedItemLink according to th...
bool SetPickedItemGroupId(KIID aId, unsigned aIdx)
Set the group id associated to a given picked item.
EDA_ITEM * GetPickedItem(unsigned int aIdx) const
A holder to handle a list of undo (or redo) commands.
std::vector< PICKED_ITEMS_LIST * > m_CommandsList
#define UR_TRANSIENT
indicates the item is owned by the undo/redo stack
This file contains miscellaneous commonly used macros and functions.
#define KI_FALLTHROUGH
The KI_FALLTHROUGH macro is to be used when switch statement cases should purposely fallthrough from ...
Class to handle a set of BOARD_ITEMs.
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
@ PCB_ZONE_T
class ZONE, a copper pour area
@ PCB_FOOTPRINT_T
class FOOTPRINT, a footprint
@ PCB_PAD_T
class PAD, a pad in a footprint
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
@ PCB_NETINFO_T
class NETINFO_ITEM, a description of a net
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
UNDO_REDO
Undo Redo considerations: Basically we have 3 cases New item Deleted item Modified item there is also...