]> git.saurik.com Git - wxWidgets.git/blame - src/common/cmdproc.cpp
No changes, just refactor common code in wxImage cloning functions.
[wxWidgets.git] / src / common / cmdproc.cpp
CommitLineData
70050c82
VZ
1///////////////////////////////////////////////////////////////////////////////
2// Name: src/common/cmdproc.cpp
3// Purpose: wxCommand and wxCommandProcessor classes
4// Author: Julian Smart (extracted from docview.h by VZ)
5// Modified by:
6// Created: 05.11.00
7// RCS-ID: $Id$
77ffb593 8// Copyright: (c) wxWidgets team
65571936 9// Licence: wxWindows licence
70050c82
VZ
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
70050c82
VZ
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#ifndef WX_PRECOMP
28 #include "wx/intl.h"
29 #include "wx/string.h"
30 #include "wx/menu.h"
62b50655 31 #include "wx/accel.h"
70050c82
VZ
32#endif //WX_PRECOMP
33
34#include "wx/cmdproc.h"
35
36// ============================================================================
37// implementation
38// ============================================================================
39
40IMPLEMENT_CLASS(wxCommand, wxObject)
41IMPLEMENT_DYNAMIC_CLASS(wxCommandProcessor, wxObject)
42
43// ----------------------------------------------------------------------------
44// wxCommand
45// ----------------------------------------------------------------------------
46
47wxCommand::wxCommand(bool canUndoIt, const wxString& name)
48{
49 m_canUndo = canUndoIt;
50 m_commandName = name;
51}
52
70050c82
VZ
53// ----------------------------------------------------------------------------
54// Command processor
55// ----------------------------------------------------------------------------
56
57wxCommandProcessor::wxCommandProcessor(int maxCommands)
58{
59 m_maxNoCommands = maxCommands;
70050c82 60#if wxUSE_MENUS
d3b9f782 61 m_commandEditMenu = NULL;
70050c82 62#endif // wxUSE_MENUS
d98c269b
VZ
63
64#if wxUSE_ACCEL
3c8b2423
VZ
65 m_undoAccelerator = '\t' + wxAcceleratorEntry(wxACCEL_CTRL, 'Z').ToString();
66 m_redoAccelerator = '\t' + wxAcceleratorEntry(wxACCEL_CTRL, 'Y').ToString();
d98c269b 67#endif // wxUSE_ACCEL
f260c476
VZ
68
69 m_lastSavedCommand =
70 m_currentCommand = wxList::compatibility_iterator();
70050c82
VZ
71}
72
73wxCommandProcessor::~wxCommandProcessor()
74{
75 ClearCommands();
76}
77
78bool wxCommandProcessor::DoCommand(wxCommand& cmd)
79{
80 return cmd.Do();
81}
82
83bool wxCommandProcessor::UndoCommand(wxCommand& cmd)
84{
85 return cmd.Undo();
86}
87
88// Pass a command to the processor. The processor calls Do();
89// if successful, is appended to the command history unless
c9d59ee7 90// storeIt is false.
70050c82
VZ
91bool wxCommandProcessor::Submit(wxCommand *command, bool storeIt)
92{
9a83f860 93 wxCHECK_MSG( command, false, wxT("no command in wxCommandProcessor::Submit") );
70050c82
VZ
94
95 if ( !DoCommand(*command) )
ca1b5af1
VZ
96 {
97 // the user code expects the command to be deleted anyhow
98 delete command;
99
c9d59ee7 100 return false;
ca1b5af1 101 }
70050c82
VZ
102
103 if ( storeIt )
104 Store(command);
f530fa27
VZ
105 else
106 delete command;
70050c82 107
c9d59ee7 108 return true;
70050c82
VZ
109}
110
111void wxCommandProcessor::Store(wxCommand *command)
112{
9a83f860 113 wxCHECK_RET( command, wxT("no command in wxCommandProcessor::Store") );
70050c82 114
70050c82
VZ
115 // Correct a bug: we must chop off the current 'branch'
116 // so that we're at the end of the command list.
117 if (!m_currentCommand)
118 ClearCommands();
119 else
120 {
222ed1d6 121 wxList::compatibility_iterator node = m_currentCommand->GetNext();
70050c82
VZ
122 while (node)
123 {
222ed1d6 124 wxList::compatibility_iterator next = node->GetNext();
b1d4dd7a 125 delete (wxCommand *)node->GetData();
222ed1d6 126 m_commands.Erase(node);
f260c476
VZ
127
128 // Make sure m_lastSavedCommand won't point to freed memory
129 if ( m_lastSavedCommand == node )
130 m_lastSavedCommand = wxList::compatibility_iterator();
131
70050c82
VZ
132 node = next;
133 }
134 }
135
1544ba0e
VZ
136 if ( (int)m_commands.GetCount() == m_maxNoCommands )
137 {
138 wxList::compatibility_iterator firstNode = m_commands.GetFirst();
139 wxCommand *firstCommand = (wxCommand *)firstNode->GetData();
140 delete firstCommand;
141 m_commands.Erase(firstNode);
142
143 // Make sure m_lastSavedCommand won't point to freed memory
144 if ( m_lastSavedCommand == firstNode )
145 m_lastSavedCommand = wxList::compatibility_iterator();
146 }
147
70050c82 148 m_commands.Append(command);
b1d4dd7a 149 m_currentCommand = m_commands.GetLast();
70050c82
VZ
150 SetMenuStrings();
151}
152
153bool wxCommandProcessor::Undo()
154{
155 wxCommand *command = GetCurrentCommand();
156 if ( command && command->CanUndo() )
157 {
158 if ( UndoCommand(*command) )
159 {
b1d4dd7a 160 m_currentCommand = m_currentCommand->GetPrevious();
70050c82 161 SetMenuStrings();
c9d59ee7 162 return true;
70050c82
VZ
163 }
164 }
165
c9d59ee7 166 return false;
70050c82
VZ
167}
168
169bool wxCommandProcessor::Redo()
170{
d3b9f782 171 wxCommand *redoCommand = NULL;
4ed40684
VZ
172 wxList::compatibility_iterator redoNode
173#if !wxUSE_STL
174 = NULL // just to avoid warnings
175#endif // !wxUSE_STL
176 ;
17271c79
VZ
177
178 if ( m_currentCommand )
70050c82 179 {
17271c79 180 // is there anything to redo?
b1d4dd7a 181 if ( m_currentCommand->GetNext() )
17271c79 182 {
b1d4dd7a
RL
183 redoCommand = (wxCommand *)m_currentCommand->GetNext()->GetData();
184 redoNode = m_currentCommand->GetNext();
17271c79 185 }
70050c82 186 }
17271c79 187 else // no current command, redo the first one
70050c82 188 {
b1d4dd7a 189 if (m_commands.GetCount() > 0)
70050c82 190 {
b1d4dd7a
RL
191 redoCommand = (wxCommand *)m_commands.GetFirst()->GetData();
192 redoNode = m_commands.GetFirst();
70050c82
VZ
193 }
194 }
195
196 if (redoCommand)
197 {
198 bool success = DoCommand(*redoCommand);
199 if (success)
200 {
201 m_currentCommand = redoNode;
202 SetMenuStrings();
c9d59ee7 203 return true;
70050c82
VZ
204 }
205 }
c9d59ee7 206 return false;
70050c82
VZ
207}
208
209bool wxCommandProcessor::CanUndo() const
210{
211 wxCommand *command = GetCurrentCommand();
212
213 return command && command->CanUndo();
214}
215
216bool wxCommandProcessor::CanRedo() const
217{
222ed1d6 218 if (m_currentCommand && !m_currentCommand->GetNext())
c9d59ee7 219 return false;
70050c82 220
222ed1d6 221 if (m_currentCommand && m_currentCommand->GetNext())
c9d59ee7 222 return true;
70050c82 223
222ed1d6 224 if (!m_currentCommand && (m_commands.GetCount() > 0))
c9d59ee7 225 return true;
70050c82 226
c9d59ee7 227 return false;
70050c82
VZ
228}
229
230void wxCommandProcessor::Initialize()
231{
b1d4dd7a 232 m_currentCommand = m_commands.GetLast();
70050c82
VZ
233 SetMenuStrings();
234}
235
236void wxCommandProcessor::SetMenuStrings()
237{
238#if wxUSE_MENUS
239 if (m_commandEditMenu)
240 {
d863ed83
JS
241 wxString undoLabel = GetUndoMenuLabel();
242 wxString redoLabel = GetRedoMenuLabel();
c9d59ee7 243
d863ed83
JS
244 m_commandEditMenu->SetLabel(wxID_UNDO, undoLabel);
245 m_commandEditMenu->Enable(wxID_UNDO, CanUndo());
246
247 m_commandEditMenu->SetLabel(wxID_REDO, redoLabel);
248 m_commandEditMenu->Enable(wxID_REDO, CanRedo());
249 }
250#endif // wxUSE_MENUS
251}
252
253// Gets the current Undo menu label.
254wxString wxCommandProcessor::GetUndoMenuLabel() const
255{
256 wxString buf;
257 if (m_currentCommand)
258 {
b1d4dd7a 259 wxCommand *command = (wxCommand *)m_currentCommand->GetData();
d863ed83 260 wxString commandName(command->GetName());
b494c48b 261 if (commandName.empty()) commandName = _("Unnamed command");
d863ed83
JS
262 bool canUndo = command->CanUndo();
263 if (canUndo)
264 buf = wxString(_("&Undo ")) + commandName + m_undoAccelerator;
265 else
266 buf = wxString(_("Can't &Undo ")) + commandName + m_undoAccelerator;
267 }
268 else
269 {
270 buf = _("&Undo") + m_undoAccelerator;
271 }
c9d59ee7 272
d863ed83
JS
273 return buf;
274}
275
276// Gets the current Undo menu label.
277wxString wxCommandProcessor::GetRedoMenuLabel() const
278{
279 wxString buf;
280 if (m_currentCommand)
281 {
282 // We can redo, if we're not at the end of the history.
b1d4dd7a 283 if (m_currentCommand->GetNext())
70050c82 284 {
b1d4dd7a 285 wxCommand *redoCommand = (wxCommand *)m_currentCommand->GetNext()->GetData();
d863ed83 286 wxString redoCommandName(redoCommand->GetName());
b494c48b 287 if (redoCommandName.empty()) redoCommandName = _("Unnamed command");
d863ed83 288 buf = wxString(_("&Redo ")) + redoCommandName + m_redoAccelerator;
70050c82
VZ
289 }
290 else
291 {
d863ed83 292 buf = _("&Redo") + m_redoAccelerator;
70050c82
VZ
293 }
294 }
d863ed83
JS
295 else
296 {
b1d4dd7a 297 if (m_commands.GetCount() == 0)
d863ed83
JS
298 {
299 buf = _("&Redo") + m_redoAccelerator;
300 }
301 else
302 {
303 // currentCommand is NULL but there are commands: this means that
304 // we've undone to the start of the list, but can redo the first.
b1d4dd7a 305 wxCommand *redoCommand = (wxCommand *)m_commands.GetFirst()->GetData();
d863ed83 306 wxString redoCommandName(redoCommand->GetName());
b494c48b 307 if (redoCommandName.empty()) redoCommandName = _("Unnamed command");
d863ed83
JS
308 buf = wxString(_("&Redo ")) + redoCommandName + m_redoAccelerator;
309 }
310 }
311 return buf;
70050c82
VZ
312}
313
314void wxCommandProcessor::ClearCommands()
315{
222ed1d6 316 wxList::compatibility_iterator node = m_commands.GetFirst();
70050c82
VZ
317 while (node)
318 {
b1d4dd7a 319 wxCommand *command = (wxCommand *)node->GetData();
70050c82 320 delete command;
222ed1d6 321 m_commands.Erase(node);
b1d4dd7a 322 node = m_commands.GetFirst();
70050c82 323 }
f260c476 324
222ed1d6 325 m_currentCommand = wxList::compatibility_iterator();
f260c476 326 m_lastSavedCommand = wxList::compatibility_iterator();
70050c82
VZ
327}
328
329