]> git.sur5r.net Git - c128-kasse/blob - src/kasse.c
log the remaining credits
[c128-kasse] / src / kasse.c
1 /* 
2  * RGB2R-C128-Kassenprogramm
3  * (c) 2007-2008 phil_fry, sECuRE, sur5r
4  * See LICENSE for license information
5  *
6  */
7 #define _IS_KASSE
8 #include <stdio.h>
9 #include <conio.h>
10 #include <stdlib.h>
11 #include <string.h>
12 #include <cbm.h>
13
14 #include "general.h"
15 #include "config.h"
16 #include "kasse.h"
17 #include "credit_manager.h"
18 #include "c128time.h"
19 // drucker 4 oder 5
20 // graphic 4,0,10
21
22 /* NOTE: undocumented function which scratches files
23    We need to use this function because linking unistd.h
24    makes our program break at runtime.
25  */
26 unsigned char __fastcall__ _sysremove(const char *name);
27
28 static void sane_exit() {
29         save_items();
30         save_credits();
31         exit(1);
32 }
33
34 /* Hauptbildschirm ausgeben */
35 static void print_screen() {
36         BYTE i = 0;
37         char *time = get_time();
38         char profit[10];
39         clrscr();
40         if (format_euro(profit, 10, money) == NULL) {
41                 cprintf("Einnahme %ld konnte nicht umgerechnet werden\r\n", money);
42                 exit(1);
43         }
44         cprintf("C128-Kassenprogramm (phil_fry, sECuRE, sur5r)\r\
45 \r\nUhrzeit: %s (wird nicht aktualisiert)\r\
46 Eingenommen: %s, Verkauft: %ld Flaschen, Drucken: %s\r\n\r\n", 
47         time, profit, items_sold, (printing == 1 ? "ein" : "aus"));
48         for (; i < status.num_items; ++i)
49                 cprintf("Eintrag %x: %s (%d Cents, %d mal verkauft)\r\n",
50                         i, status.status[i].item_name, status.status[i].price, status.status[i].times_sold);
51         cprintf("\r\nBefehle: s) Daten sichern d) Drucken umschalten\r\
52 g) Guthabenverwaltung     z) Zeit setzen\r\
53 f) Freitext verkaufen     q) Beenden\r\n");
54 }
55
56 static void log_file(const char *s) {
57         /* A log-entry has usually 50 bytes, so we take 64 bytes.
58            Because files are wrapped (log.0, log.1, ...) every 100
59            lines, we don't need more than 100 * 64 bytes. */
60         char *buffer = malloc(sizeof(char) * 64 * 100);
61         char filename[8];
62         int read = 0;
63         unsigned int c;
64         if (buffer == NULL) {
65                 cprintf("No memory available\n");
66                 }
67         buffer[0] = '\0';
68         if (((++log_lines_written) % 100) == 0)
69                 log_num++;
70         sprintf(filename, "log-%d", log_num);
71         /* Don't read log if there were no lines written before */
72         if (log_lines_written != 1) {
73                 if ((c = cbm_open((BYTE)8, (BYTE)8, (BYTE)0, filename)) != 0) {
74                         c128_perror(c, "cbm_open(log)");
75                         sane_exit();
76                 }
77                 read = cbm_read((BYTE)8, buffer, sizeof(char) * 64 * 100);
78                 cbm_close((BYTE)8);
79                 _sysremove(filename);
80         }
81         if ((c = cbm_open((BYTE)8, (BYTE)8, (BYTE)1, filename)) != 0) {
82                 c128_perror(c, "cbm_open(log)");
83                 sane_exit();
84         }
85         if (read < 0) {
86                 cprintf("Could not read existing logfile (read returned %d)\n", read);
87                 sane_exit();
88         }
89         strcpy(buffer+read, s);
90         c = cbm_write((BYTE)8, buffer, read+strlen(s));
91         if (c != (read+strlen(s))) {
92                 cprintf("Could not save logfile (wrote %d bytes, wanted %d bytes), please make sure the floppy is not full!\n", c, (read+strlen(s)));
93                 sane_exit();
94         }
95         cbm_close((BYTE)8);
96         free(buffer);
97 }
98
99 static char retry_or_quit() {
100         char *c;
101         do {
102                 cprintf("\r\nr)etry or q)uit?\r\n");
103                 c = get_input();
104         } while ((*c != 'r') && (*c != 'q'));
105         return *c;
106 }
107
108 /* Prints a line and logs it to file */
109 static void print_log(char *name, int item_price, int einheiten, char *nickname, char *rest) {
110         BYTE c;
111         char *time = get_time();
112         char price[10];
113         /* Format: 
114            Transaction-ID (Anzahl verkaufter Einträge, inklusive des zu druckenden!) -- 6-stellig
115            Uhrzeit -- 8-stellig
116            Eintragname (= Getränk) -- 9-stellig
117            Preis (in Cents) -- 7-stellig
118            Anzahl -- 2-stellig
119            Nickname (falls es vom Guthaben abgezogen wird) -- 10-stellig
120            restguthaben (9-stellig)
121
122            + 7 leerzeichen
123            --> 48 zeichen
124            */
125         if (format_euro(price, 10, item_price) == NULL) {
126                 cprintf("Preis %d konnte nicht umgerechnet werden\r\n", item_price);
127                 exit(1);
128         }
129
130         sprintf(print_buffer, "%c[%lu] %s - %-9s - %s - r %s - %d - an %s\r",  17,
131                         items_sold, time, name, price, rest,
132                         einheiten, (*nickname != '\0' ? nickname : "Unbekannt"));
133 RETRY:
134         c = cbm_open((BYTE)4, (BYTE)4, (BYTE)0, NULL);
135         if (c != 0) {
136                 c128_perror(c, "cbm_open(printer)");
137                 if (retry_or_quit() == 'q')
138                         sane_exit();
139
140                 goto RETRY;
141         }
142         c = cbm_write((BYTE)4, print_buffer, strlen(print_buffer));
143         if (c != strlen(print_buffer)) {
144                 c128_perror(c, "write(printer)");
145                 if (retry_or_quit() == 'q') {
146                         save_items();
147                         save_credits();
148                         exit(1);
149                 }
150                 goto RETRY;
151         }
152         cbm_close((BYTE)4);
153         log_file(print_buffer);
154 }
155
156 /* dialog which is called for each bought item */
157 BYTE buy(char *name, unsigned int price) {
158         int negative = 1;
159         char entered[5] = {'1', 0, 0, 0, 0};
160         BYTE i = 0, matches = 0;
161         BYTE c, nickname_len;
162         int einheiten;
163         char *input;
164         char nickname[11];
165         char rest[9];
166         struct credits_t *credit;
167
168         memset(rest, ' ', sizeof(rest));
169         rest[8] = '\0';
170
171         cprintf("Wieviel Einheiten \"%s\"? [1] \r\n", name);
172         while (1) {
173                 c = getchar();
174                 if (c == 13)
175                         break;
176                 else if (c == 27) {
177                         cprintf("Kauf abgebrochen, druecke RETURN...\r\n");
178                         get_input();
179                         return 1;
180                 } else if (c == '-' && i == 0)
181                         negative = -1;
182                 else if (c > 47 && c < 58)
183                         entered[i++] = c;
184         }
185         einheiten = atoi(entered) * negative;
186         
187         toggle_videomode();
188         cprintf("%dx %s fuer ", einheiten, name);
189         toggle_videomode();
190         
191         cprintf("\r\nAuf ein Guthaben kaufen? Wenn ja, Nickname eingeben:\r\n");
192         input = get_input();
193         strncpy(nickname, input, 11);
194         if (*nickname != '\0') {
195                 toggle_videomode();
196                 cprintf("%s\r\n", nickname);
197                 toggle_videomode();
198         }
199
200         if (nickname != NULL && *nickname != '\0' && *nickname != 32) {
201                 nickname_len = strlen(nickname);
202                 /* go through credits and remove the amount of money or set nickname
203                  * to NULL if no such credit could be found */
204                 credit = find_credit(nickname);
205                 if (credit != NULL) {
206                         if ((signed int)credit->credit < ((signed int)price * einheiten)) {
207                                 cprintf("Sorry, %s hat nicht genug Geld :-(\r\n", nickname);
208                                 get_input();
209                                 return 0;
210                         }
211                         /* substract money */
212                         credit->credit -= (price * einheiten);
213
214                         if (format_euro(rest, 10, credit->credit) == NULL) {
215                                 cprintf("Preis %d konnte nicht umgerechnet werden\r\n", credit->credit);
216                                 exit(1);
217                         }
218
219                         cprintf("\r\nVerbleibendes Guthaben fuer %s: %s. Druecke RETURN...\r\n",
220                                 nickname, rest);
221                         toggle_videomode();
222                         cprintf("\r\nDein verbleibendes Guthaben betraegt %s.\r\n", rest);
223                         toggle_videomode();
224                         get_input();
225                         matches++;
226                 } else {
227                         cprintf("\r\nNickname nicht gefunden in der Guthabenverwaltung! Abbruch, druecke RETURN...\r\n");
228                         get_input();
229                         return 0;
230                 }
231         } else {
232                 /* Ensure that nickname is NULL if it's empty because it's used in print_log */
233                 *nickname = '\0';
234         }
235         
236         money += price * einheiten;
237         items_sold += einheiten;
238         if (printing == 1)
239                 print_log(name, price, einheiten, nickname, rest);
240
241         return einheiten;
242 }
243
244 void buy_stock(BYTE n) {
245         if (n >= status.num_items || status.status[n].item_name == NULL) {
246                 cprintf("FEHLER: Diese Einheit existiert nicht.\r\n");
247                 get_input();
248                 return;
249         }
250
251         status.status[n].times_sold += buy(status.status[n].item_name, status.status[n].price);
252 }
253
254 void buy_custom() {
255         BYTE c = 0, i = 0;
256         int negative = 1;
257         char entered[5] = {'1', 0, 0, 0, 0};
258         char *input, name[20];
259         int price;
260
261         memset(name, '\0', 20);
262         cprintf("\r\nWas soll gekauft werden?\r\n");
263         input = get_input();
264         strncpy(name, input, 20);
265         if (*name == '\0')
266                 return;
267
268         cprintf("\r\nWie teuer ist \"%s\" (in cents)?\r\n", name);
269         while (1) {
270                 c = getchar();
271                 if (c == 13)
272                         break;
273                 else if (c == 27) {
274                         cprintf("Kauf abgebrochen, druecke RETURN...\r\n");
275                         get_input();
276                         return;
277                 } else if (c == '-' && i == 0)
278                         negative = -1;
279                 else if (c > 47 && c < 58)
280                         entered[i++] = c;
281         }
282         price = atoi(entered) * negative;
283
284         cprintf("\r\n");
285
286         buy(name, price);
287 }
288
289 void set_time_interactive() {
290         BYTE part[3] = {'0', '0', '\0'};
291         BYTE tp1, tp2, tp3;
292         char *time_input, *time;
293         cprintf("Gib die aktuelle Uhrzeit ein (Format HHMMSS):\r\n");
294         time_input = get_input();
295         part[0] = time_input[0];
296         part[1] = time_input[1];
297         tp1 = atoi(part);
298         part[0] = time_input[2];
299         part[1] = time_input[3];
300         tp2 = atoi(part);
301         part[0] = time_input[4];
302         part[1] = time_input[5];
303         tp3 = atoi(part);
304         set_time(tp1, tp2, tp3);
305
306         time = get_time();
307         cprintf("\r\nZeit gesetzt: %s\r\n", time);
308 }
309
310 int main() {
311         char *c;
312
313         if (VIDEOMODE == 40)
314                 toggle_videomode();
315         /* Set time initially, c128 doesn't know it */
316         set_time_interactive();
317
318         POKE(216, 255);
319
320         /* Load configuration */
321         load_config();
322         cprintf("got %d logfiles\r\n", log_num);
323
324         /* Load items (= drinks) */
325         load_items();
326         /* Load credits */
327         load_credits();
328         while (1) {
329                 print_screen();
330                 c = get_input();
331                 /* ...display dialogs eventually */
332                 if (*c > 47 && *c < 58) {
333                         buy_stock((*c) - 48);
334                         toggle_videomode();
335                         clrscr();
336                         toggle_videomode();
337                 } else if (*c == 'f') {
338                         buy_custom();
339                         toggle_videomode();
340                         clrscr();
341                         toggle_videomode();
342                 } else if (*c == 's') {
343                         save_items();
344                         save_credits();
345                         cprintf("Statefile/Creditfile gesichert, druecke RETURN...\r\n");
346                         get_input();
347                 } else if (*c == 'd') {
348                         /* enable/disable printing */
349                         printing = (printing == 1 ? 0 : 1);
350                         cprintf("Drucken ist nun %s, druecke RETURN...\r\n", 
351                                 (printing == 1 ? "eingeschaltet" : "ausgeschaltet"));
352                         get_input();
353                 } else if (*c == 'g') {
354                         credit_manager();
355                 } else if (*c == 'z') {
356                         set_time_interactive();
357                 } else if (*c == 'q')
358                         break;
359         }
360         cprintf("BYEBYE\r\n");
361
362         return 0;
363 }