-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsudoku_undo.c
325 lines (287 loc) · 11.1 KB
/
sudoku_undo.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
/******************************************************************************
* Program: sudoku.c
*
* Purpose: To implement a history system that keeps track of changes made to
* a sudoku board over time. Allows program to undo or redo steps.
*
* Developer: Philip Ormand
*
* Date: 5/13/16
*
*****************************************************************************/
#include <stdlib.h>
#include "sudoku_board.h"
#include "sudoku_undo.h"
#include "sudoku_utility.h"
// internal structs defined in source file, not header file, for information-hiding
// purposes. "clients" of the History ADT don't need to know how it's implemented,
// only the functions they can call to make use of its functionality
/**
* Struct representing the history of a sudoku board
*/
struct HistoryStruct {
HistoryStep *steps; /**< dynamically allocated array of HistorySteps */
HistoryStep *currentStep; /**< off-the-end pointer to top of History stack */
size_t length; /**< number of steps currently in history */
size_t capacity; /**< number of elements allocated in 'steps' */
struct SudokuBoard *owner; /**< whose history is this? So we don't have to pass
it as a parameter to any "member" functions */
};
typedef struct HistoryStruct HistoryStruct;
/**
* Factory function that creates and initializes a dynamic History object for a
* SudokuBoard. The SudokuBoard calls this function with itself as a parameter
* and stores the return value in its history member.
*
* @param owner SudokuBoard that will own the created History stack
*/
History createHistory(struct SudokuBoard *owner)
{
// allocate space for a HistoryStruct
History history = malloc(sizeof(*history));
if (history == NULL) // failed to allocate HistoryStruct
{
terminate("ERROR: could not create History");
}
// allocate a dynamic array for holding HistorySteps
history->capacity = 1;
history->length = 0;
history->steps = malloc(sizeof(HistoryStep) * history->capacity);
if (history->steps == NULL)
{
terminate("ERROR: could not initialize History steps");
}
// position currentStep to be ready for recording the first history event
history->currentStep = history->steps;
// record owner of this History stack for future undo/redo operations
history->owner = owner;
return history;
}
/**
* Frees a dynamically-allocated HistoryStruct to prevent memory leaks.
* Pointer passed to function will be set to NULL at the end, indicating that it was freed.
*
* @param historyPtr Pointer to HistoryStruct to be freed
*/
void freeHistory(History *historyPtr)
{
if (historyPtr && *historyPtr)
{
History history = *historyPtr;
// deallocate memory used to store all the HistorySteps
free(history->steps);
// deallocate the History object itself
free(history);
// make pointer null, so it can't be accidently freed again
*historyPtr = NULL;
}
else
{
terminate("ERROR: tried to free a null History pointer");
}
}
/**
* Reallocates a HistoryStruct's 'steps' member with a larger size if there not enough room to add
* the specified number of HistorySteps
* No change occurs if capacity is large enough already.
*
* @param history Pointer to HistoryStruct that will be modified
* @param stepsToAdd Number of additional HistorySteps to ensure space for
*/
void growHistory(History history, size_t stepsToAdd)
{
if (history)
{
if (history->steps)
{
if (stepsToAdd > (history->capacity - history->length))
{
int currentStepIndex = history->currentStep - history->steps;
history->capacity *= 2;
history->steps = realloc(history->steps, sizeof(HistoryStep) * history->capacity);
if (history->steps == NULL)
{
terminate("ERROR: unable to grow history");
}
history->currentStep = history->steps + currentStepIndex;
}
}
else
{
terminate("ERROR: tried to grow a History with null steps array");
}
}
else
{
terminate("ERROR: tried to grow a null History");
}
}
/**
* Add a HistoryStep to a HistoryStruct stack
*
* @param history Pointer to HistoryStruct that will receive step
* @param square Row/column address of the change
* @param value New value for the square
*/
void addUndoStep(History history, Coord2D *square, int value)
{
if (history)
{
if (validateSudokuCoord2D(square))
{
if (validateSudokuDigit(value))
{
// store location and current value of sudoku square
history->currentStep->location = *square;
history->currentStep->newValue = value;
history->currentStep->oldValue =
history->owner->contents[square->row][square->col];
// advance history
++history->currentStep;
++history->length;
growHistory(history, 1);
}
else
{
terminate("ERROR: tried to add undo step with invalid sudoku digit");
}
}
else
{
terminate("ERROR: tried to add undo step with invalid square address");
}
}
else
{
terminate("ERROR: tried to add undo step to null History");
}
}
/**
* Roll back the changes associated with the most recent HistoryStep(s) in a HistoryStruct
* Automatically recognizes if bottom of the history stack is reached.
*
* @param history Pointer to HistoryStruct
* @param stepsToUndo Number of HistorySteps to roll back
* @return True if any steps were undone, false if no steps were undone
*/
bool undoStep(History history, size_t stepsToUndo)
{
bool status = true;
if (history)
{
if (history->currentStep)
{
// only take action if the number of steps is not 0
if (stepsToUndo)
{
// if currentStep is the bottom of the stack, there are no steps to undo
if (history->currentStep == history->steps)
{
puts("<no steps to undo>");
status = false;
}
else
{
Coord2D *currentSquare = &(history->currentStep - 1)->location;
size_t i, existingValue;
printf("Undoing %d steps:\n", stepsToUndo);
for (i = 0; i < stepsToUndo && history->currentStep != history->steps; ++i)
{
--history->currentStep;
currentSquare = &history->currentStep->location;
existingValue = history->owner->contents[currentSquare->row][currentSquare->col];
history->owner->contents[currentSquare->row][currentSquare->col] =
history->currentStep->oldValue;
printf(" %d: changed %c%d from %d back to %d\n", i + 1,
colLabels[currentSquare->col], currentSquare->row + 1,
existingValue, history->currentStep->oldValue);
}
// if function was asked to undo more steps than were available
if (i < stepsToUndo)
{
puts("<no more steps to undo>");
}
}
}
}
else
{
terminate("ERROR: can't restore undo step if currentStep is null");
}
}
else
{
terminate("ERROR: tried to restore undo step from a null History");
}
return status;
}
/**
* Reapplies the changes associated with the most recent undone HistoryStep(s) in a HistoryStruct
* Automatically recognizes when the top of history stack is reached.
*
* @param history Pointer to HistoryStruct
* @param stepsToUndo Number of HistorySteps to reapply
* @return True if any steps were redone, false if no steps were redone
*/
bool redoStep(History history, size_t stepsToRedo)
{
bool status = true;
if (history)
{
if (history->currentStep)
{
// only take action if the number of steps is not 0
if (stepsToRedo)
{
// if currentStep is the top of the stack, there are no steps to redo
if (history->currentStep == history->steps + history->length)
{
puts("<no steps to redo>");
status = false;
}
else
{
Coord2D *currentSquare = &(history->currentStep - 1)->location;
size_t i, existingValue;
printf("Redoing %d steps:\n", stepsToRedo);
for (i = 0; i < stepsToRedo && history->currentStep != history->steps + history->length; ++i)
{
currentSquare = &history->currentStep->location;
existingValue = history->owner->contents[currentSquare->row][currentSquare->col];
history->owner->contents[currentSquare->row][currentSquare->col] =
history->currentStep->newValue;
printf(" %d: changed %c%d from %d back to %d\n", i + 1,
colLabels[currentSquare->col], currentSquare->row + 1,
existingValue, history->currentStep->newValue);
++history->currentStep;
}
// if function was asked to undo more steps than were available
if (i < stepsToRedo)
{
puts("<no more steps to redo>");
}
}
}
}
else
{
terminate("ERROR: can't restore undo step if currentStep is null");
}
}
else
{
terminate("ERROR: tried to restore undo step from a null History");
}
return status;
}
/**
* If there are any HistorySteps PAST the current history location (i.e. steps that could be redone),
* make them inaccessible, because they are no longer valid.
*
* @param history Pointer to HistoryStruct
*/
void invalidateSubsequentRedoSteps(History history)
{
// set current state of board as final available redo step
history->length = history->currentStep - history->steps;
}