home · contact · privacy
Applied new safer file saving method to record file in turn_over(), too, added error...
[plomrogue] / src / misc.c
1 /* misc.c */
2
3 #include "misc.h"
4 #include <stdio.h> /* for rename() */
5 #include <unistd.h> /* for unlink(), acess() */
6 #include <stdlib.h> /* for calloc(), free() */
7 #include <string.h> /* for strlen(), strcmp(), memcpy() */
8 #include <stdint.h> /* for uint8_t */
9 #include "windows.h" /* for suspend_win(), append_win(), reset_pad_offset(),
10                       * resize_active_win(), cycle_active_win(),
11                       * shift_active_win(), struct Win, struct WinMeta
12                       */
13 #include "keybindings.h" /* for get_action_key(), save_keybindings(),
14                           * keyswin_move_selection(), keyswin_mod_key()
15                           */
16 #include "readwrite.h" /* for [read/write]_uint[8/16/32][_bigendian]() */
17 #include "map_objects.h" /* for struct Monster, write_map_objects(), */
18 #include "map_object_actions.h" /* for is_passable(), move_monster() */
19 #include "map.h" /* for map_scroll(),map_center_player(), Map struct,dir enum */
20 #include "main.h" /* for World struct */
21 #include "yx_uint16.h" /* for yx_uint16 */
22 #include "rrand.h" /* for rrand(), rrand_seed() */
23 #include "rexit.h" /* for exit_err() */
24
25
26 extern void textfile_sizes(FILE * file, uint16_t * linemax_p,
27                            uint16_t * n_lines_p)
28 {
29     uint16_t n_lines = 0;
30     int c = 0;
31     uint16_t linemax = 0;
32     uint16_t c_count = 0;
33     while (EOF != c)
34     {
35         c_count++;
36         c = getc(file);
37         if ('\n' == c)
38         {
39             if (c_count > linemax)
40             {
41                 linemax = c_count + 1;
42             }
43             c_count = 0;
44             if (n_lines_p)
45             {
46                 n_lines++;
47             }
48         }
49     }
50     fseek(file, 0, SEEK_SET);
51     * linemax_p = linemax;
52     if (n_lines_p)
53     {
54         * n_lines_p = n_lines;
55     }
56 }
57
58
59
60 extern void update_log(struct World * world, char * text)
61 {
62     static char * last_msg;
63     if (0 == last_msg)
64     {
65         last_msg = calloc(1, sizeof(char));
66     }
67     char * new_text;
68     uint16_t len_old = strlen(world->log);
69     if (0 == strcmp(last_msg, text))
70     {
71         uint16_t len_whole = len_old + 1;
72         new_text = calloc(len_whole + 1, sizeof(char));
73         memcpy(new_text, world->log, len_old);
74         memcpy(new_text + len_old, ".", 1);
75     }
76     else
77     {
78         uint16_t len_new = strlen(text);
79         uint16_t len_whole = len_old + len_new + 1;
80         new_text = calloc(len_whole, sizeof(char));
81         memcpy(new_text, world->log, len_old);
82         memcpy(new_text + len_old, text, len_new);
83         last_msg = calloc(len_new + 1, sizeof(char));
84         memcpy(last_msg, text, len_new);
85     }
86     free(world->log);
87     world->log = new_text;
88 }
89
90
91
92 extern uint16_t center_offset(uint16_t pos, uint16_t mapsize,
93                               uint16_t framesize)
94 {
95     uint16_t offset = 0;
96     if (mapsize > framesize)
97     {
98         if (pos > framesize / 2)
99         {
100             if (pos < mapsize - (framesize / 2))
101             {
102                 offset = pos - (framesize / 2);
103             }
104             else
105             {
106                 offset = mapsize - framesize;
107             }
108         }
109     }
110     return offset;
111 }
112
113
114
115 extern void turn_over(struct World * world, char action)
116 {
117     char * err_open  = "Error recording move: "
118                        "Unable to open 'record_tmp' for appending.";
119     char * err_write = "Error recording move: "
120                        "Trouble writing to opened 'record_tmp'.";
121     char * err_close = "Error recording move: "
122                        "Unable to close opened 'record_tmp'.";
123     char * err_unl   = "Error recording move: "
124                        "Unable to unlink old 'record' file.";
125     char * err_move  = "Error recording move: "
126                         "Unable to rename 'record_tmp' to 'record'.";
127     char * recordfile_tmp = "record_tmp";
128     char * recordfile     = "record";
129     if (1 == world->interactive)
130     {
131         FILE * file = fopen(recordfile_tmp, "a");
132         exit_err(0 == file, world, err_open);
133         exit_err(write_uint8(action, file), world, err_write);
134         exit_err(fclose(file), world, err_close);
135         exit_err(unlink(recordfile), world, err_unl);
136         exit_err(rename(recordfile_tmp, recordfile), world, err_move);
137     }
138
139     world->turn++;
140     rrand_seed(world->seed * world->turn);
141     struct Monster * monster;
142     for (monster = world->monster;
143          monster != 0;
144          monster = monster->map_obj.next)
145     {
146         move_monster(world, monster);
147     }
148 }
149
150
151
152 extern void save_game(struct World * world)
153 {
154     char * err_open  = "Error saving game: "
155                        "Unable to open 'savefile_tmp' for writing.";
156     char * err_write = "Error saving game: "
157                        "Trouble writing to opened 'savefile_tmp'.";
158     char * err_close = "Error saving game: "
159                        "Unable to close opened 'savefile_tmp'.";
160     char * err_unl   = "Error saving game: "
161                        "Unable to unlink old 'savefile'.";
162     char * err_move  = "Error saving game: "
163                         "Unable to rename 'savefile_tmp' to 'savefile'.";
164     char * savefile_tmp = "savefile_tmp";
165     char * savefile     = "savefile";
166     FILE * file = fopen(savefile_tmp, "w");
167     exit_err(0 == file, world, err_open);
168     if (   write_uint32_bigendian(world->seed, file)
169         || write_uint32_bigendian(world->turn, file)
170         || write_uint16_bigendian(world->player->pos.y + 1, file)
171         || write_uint16_bigendian(world->player->pos.x + 1, file)
172         || write_uint8(world->player->hitpoints, file)
173         || write_map_objects(world, world->monster, file)
174         || write_map_objects(world, world->item, file))
175     {
176         exit_err(1, world, err_write);
177     }
178     exit_err(fclose(file), world, err_close);
179     if (!access(savefile, F_OK))
180     {
181         exit_err(unlink(savefile), world, err_unl);
182     }
183     exit_err(rename(savefile_tmp, savefile), world, err_move);
184 }
185
186
187
188 extern void toggle_window(struct WinMeta * win_meta, struct Win * win)
189 {
190     if (0 != win->frame.curses_win)
191     {
192         suspend_win(win_meta, win);
193     }
194     else
195     {
196         append_win(win_meta, win);
197     }
198 }
199
200
201
202 extern void scroll_pad(struct WinMeta * win_meta, char dir)
203 {
204     if      ('+' == dir)
205     {
206         reset_pad_offset(win_meta, win_meta->pad_offset + 1);
207     }
208     else if ('-' == dir)
209     {
210         reset_pad_offset(win_meta, win_meta->pad_offset - 1);
211     }
212 }
213
214
215
216 extern void growshrink_active_window(struct WinMeta * win_meta, char change)
217 {
218     if (0 != win_meta->active)
219     {
220         struct yx_uint16 size = win_meta->active->frame.size;
221         if      (change == '-')
222         {
223             size.y--;
224         }
225         else if (change == '+')
226         {
227             size.y++;
228         }
229         else if (change == '_')
230         {
231             size.x--;
232         }
233         else if (change == '*')
234         {
235             size.x++;
236         }
237         resize_active_win (win_meta, size);
238     }
239 }
240
241
242
243 extern struct yx_uint16 find_passable_pos(struct Map * map)
244 {
245     struct yx_uint16 pos;
246     for (pos.y = pos.x = 0; 0 == is_passable(map, pos);)
247     {
248         pos.y = rrand() % map->size.y;
249         pos.x = rrand() % map->size.x;
250     }
251     return pos;
252 }
253
254
255
256 extern uint8_t meta_keys(int key, struct World * world,
257                          struct WinMeta * win_meta, struct Win * win_keys,
258                          struct Win * win_map, struct Win * win_info,
259                          struct Win * win_log)
260 {
261     if (key == get_action_key(world->keybindings, "quit"))
262     {
263         return 1;
264     }
265     else if (key == get_action_key(world->keybindings, "scroll pad right"))
266     {
267         scroll_pad (win_meta, '+');
268     }
269     else if (key == get_action_key(world->keybindings, "scroll pad left"))
270     {
271         scroll_pad (win_meta, '-');
272     }
273     else if (key == get_action_key(world->keybindings, "toggle keys window"))
274     {
275         toggle_window(win_meta, win_keys);
276     }
277     else if (key == get_action_key(world->keybindings, "toggle map window"))
278     {
279         toggle_window(win_meta, win_map);
280     }
281     else if (key == get_action_key(world->keybindings, "toggle info window"))
282     {
283         toggle_window(win_meta, win_info);
284     }
285     else if (key == get_action_key(world->keybindings, "toggle log window"))
286     {
287         toggle_window(win_meta, win_log);
288     }
289     else if (key == get_action_key(world->keybindings, "cycle forwards"))
290     {
291         cycle_active_win(win_meta, 'n');
292     }
293     else if (key == get_action_key(world->keybindings, "cycle backwards"))
294     {
295         cycle_active_win(win_meta, 'p');
296     }
297     else if (key == get_action_key(world->keybindings, "shift forwards"))
298     {
299         shift_active_win(win_meta, 'f');
300     }
301     else if (key == get_action_key(world->keybindings, "shift backwards"))
302     {
303         shift_active_win(win_meta, 'b');
304     }
305     else if (key == get_action_key(world->keybindings, "grow horizontally"))
306     {
307         growshrink_active_window(win_meta, '*');
308     }
309     else if (key == get_action_key(world->keybindings, "shrink horizontally"))
310     {
311         growshrink_active_window(win_meta, '_');
312     }
313     else if (key == get_action_key(world->keybindings, "grow vertically"))
314     {
315         growshrink_active_window(win_meta, '+');
316     }
317     else if (key == get_action_key(world->keybindings, "shrink vertically"))
318     {
319         growshrink_active_window(win_meta, '-');
320     }
321     else if (key == get_action_key(world->keybindings, "save keys"))
322     {
323         save_keybindings(world);
324     }
325     else if (key == get_action_key(world->keybindings, "keys nav up"))
326     {
327         keyswin_move_selection (world, 'u');
328     }
329     else if (key == get_action_key(world->keybindings, "keys nav down"))
330     {
331         keyswin_move_selection (world, 'd');
332     }
333     else if (key == get_action_key(world->keybindings, "keys mod"))
334     {
335         keyswin_mod_key (world, win_meta);
336     }
337     else if (key == get_action_key(world->keybindings, "map up"))
338     {
339         map_scroll (world->map, NORTH, win_map->frame.size);
340      }
341     else if (key == get_action_key(world->keybindings, "map down"))
342     {
343         map_scroll (world->map, SOUTH, win_map->frame.size);
344     }
345     else if (key == get_action_key(world->keybindings, "map right"))
346     {
347         map_scroll (world->map, EAST, win_map->frame.size);
348     }
349     else if (key == get_action_key(world->keybindings, "map left"))
350     {
351         map_scroll (world->map, WEST, win_map->frame.size);
352     }
353     else if (key == get_action_key(world->keybindings, "map center player"))
354     {
355         map_center_player (world->map, world->player, win_map->frame.size);
356     }
357     return 0;
358 }