X-Git-Url: https://git.sur5r.net/?a=blobdiff_plain;f=src%2Fkasse.c;h=d6ffb486c3ed5931eb46dd3956cb9d9b102b514f;hb=750dbde18501efb6f9520d378475e3b58aea9ea4;hp=3cb30b50eb48a4a311fcf153a281075bbd9dd5ba;hpb=fcb886f261c87269b3256f98c3177edd3fdd0bbd;p=c128-kasse diff --git a/src/kasse.c b/src/kasse.c index 3cb30b5..d6ffb48 100644 --- a/src/kasse.c +++ b/src/kasse.c @@ -1,6 +1,6 @@ -/* +/* * RGB2R-C128-Kassenprogramm - * (c) 2007 phil_fry, sECuRE, sur5r + * © 2007-2009 phil_fry, sECuRE, sur5r * See LICENSE for license information * */ @@ -9,229 +9,430 @@ #include #include #include +#include +#include <6502.h> #include "general.h" #include "config.h" #include "kasse.h" #include "credit_manager.h" #include "c128time.h" +#include "print.h" +#include "version.h" +#include "vdc_patch_charset.h" +#include "globals.h" // drucker 4 oder 5 // graphic 4,0,10 -char print_buffer[81]; +void print_item(BYTE i) { + char profit[10]; + if (format_euro(profit, sizeof(profit), status.status[i].price) == NULL) { + cprintf("Preis %ld konnte nicht umgerechnet werden\r\n", + status.status[i].price); + exit(1); + } + textcolor(TC_YELLOW); + cprintf("%2d", i); + textcolor(TC_LIGHT_GRAY); + cprintf(": %-" xstr(MAX_ITEM_NAME_LENGTH) "s \xDD%s, %3dx ", + status.status[i].item_name, profit, status.status[i].times_sold); +} /* Hauptbildschirm ausgeben */ -static void print_screen() { - BYTE i = 0; - char *time = get_time(); - char profit[10]; - clrscr(); - if (format_euro(profit, 10, money) == NULL) { - cprintf("Einname %ld konnte nicht umgerechnet werden\r\n", money); - exit(1); - } - cprintf("C128-Kassenprogramm (phil_fry, sECuRE, sur5r)\r\ -\r\nUhrzeit: %s (wird nicht aktualisiert)\r\ -Eingenommen: %s, Verkauft: %ld Flaschen, Drucken: %s\r\n\r\n", - time, profit, items_sold, (printing == 1 ? "ein" : "aus")); - for (; i < status.num_items; ++i) - cprintf("Eintrag %x: %s (%d Cents, %d mal verkauft)\r\n", - i, status.status[i].item_name, status.status[i].price, status.status[i].times_sold); - cprintf("\r\nBefehle: s) Daten sichern d) Drucken umschalten\r\ -g) Guthabenverwaltung z) Zeit setzen\r\ -n) Neues Getraenk\r\n"); +static void print_screen(void) { + BYTE i = 0; + char *time = get_time(); + char profit[10]; + clrscr(); + if (format_euro(profit, sizeof(profit), money) == NULL) { + cprintf("Einnahme %ld konnte nicht umgerechnet werden\r\n", money); + exit(1); + } + textcolor(TC_CYAN); + cprintf("C128-Kassenprogramm (phil_fry, sECuRE, sur5r, mxf) " GV "\r\n"); + textcolor(TC_LIGHT_GRAY); + cprintf("\r\nUhrzeit: %s (wird nicht aktualisiert)\r\n" + "Eingenommen: %s, Verkauft: %ld Dinge, Drucken: %s\r\n", + time, profit, items_sold, (printing == 1 ? "ein" : "aus")); + textcolor(TC_LIGHT_GRAY); + cprintf(" \xB0" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xB2" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xB2" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xB2" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xAE" + "\r\n"); + for (; i < min(status.num_items, 15); ++i) { + + cprintf(" \xDD"); + print_item(i); + cprintf("\xDD"); + + /* if we have more than 15 items, use the second column */ + if ((i + 15) < status.num_items) { + print_item(i + 15); + cprintf("\xDD"); + } else { + cprintf(" \xDD \xDD"); + } + + cprintf("\r\n"); + } + cprintf(" \xAD" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xB1" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xB1" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xB1" + "\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xC0\xBD" + "\r\n"); + textcolor(TC_YELLOW); + cprintf(" s"); + textcolor(TC_LIGHT_GRAY); + cprintf(") Daten sichern "); + textcolor(TC_YELLOW); + cprintf("g"); + textcolor(TC_LIGHT_GRAY); + cprintf(") Guthabenverwaltung\r\n"); + textcolor(TC_YELLOW); + cprintf(" z"); + textcolor(TC_LIGHT_GRAY); + cprintf(") Zeit setzen "); + textcolor(TC_YELLOW); + cprintf("f"); + textcolor(TC_LIGHT_GRAY); + cprintf(") Freitext verkaufen "); + textcolor(TC_YELLOW); + cprintf("q"); + textcolor(TC_LIGHT_GRAY); + cprintf(") Beenden\r\n"); } -static void log_file(const char *s) { - BYTE c = cbm_open((BYTE)8, (BYTE)8, (BYTE)1, "log"); - if (c != 0) { - c128_perror(c, "cbm_open(log)"); - save_items(); - save_credits(); - exit(1); - } - c = cbm_write((BYTE)8, s, strlen(s)); - if (c != strlen(s)) { - cprintf("Could not save logfile, please make sure the floppy is not full!\n"); - save_items(); - save_credits(); - exit(1); - } - cbm_close((BYTE)8); +/* + * Prints a line and logs it to file. Every line can be at max 80 characters. + * + */ +static void print_log(char *name, int item_price, int einheiten, char *nickname, + char *rest) { + char *time = get_time(); + char price[10]; + /* Format: + Transaction-ID (Anzahl verkaufter Einträge, inklusive des zu druckenden!) + -- 6-stellig + Uhrzeit -- 8-stellig + Eintragname (= Getränk) -- 9-stellig + Preis (in Cents) -- 9-stellig + Anzahl -- 2-stellig + Nickname (falls es vom Guthaben abgezogen wird) -- 10-stellig + restguthaben (9-stellig) + + + 7 leerzeichen + --> 48 zeichen + */ + if (format_euro(price, sizeof(price), item_price) == NULL) { + cprintf("Preis %d konnte nicht umgerechnet werden\r\n", item_price); + exit(1); + } + + sprintf(print_buffer, "%c[%3u] %s - %-" xstr( + MAX_ITEM_NAME_LENGTH) "s - %s - %s - %d - an %s\r", + 17, status.transaction_id, time, name, price, rest, einheiten, + (*nickname != '\0' ? nickname : "Unbekannt")); + status.transaction_id++; + print_the_buffer(); } -/* Druckt eine entsprechende Zeile aus */ -static void print_log(BYTE n, int einheiten, char *nickname) { - BYTE c; - char *time = get_time(); - char price[10]; - /* Format: - Transaction-ID (Anzahl verkaufter Einträge, inklusive des zu druckenden!) - Uhrzeit - Eintragname (= Getränk) - Preis (in Cents) - Anzahl - Nickname (falls es vom Guthaben abgezogen wird) - */ - if (format_euro(price, 10, status.status[n].price) == NULL) { - cprintf("Preis %d konnte nicht umgerechnet werden\r\n", status.status[n].price); - exit(1); - } - - sprintf(print_buffer, "[%lu] %s - %s - %s - %d - an %s\r\n", - items_sold, time, status.status[n].item_name, price, - einheiten, (nickname != NULL ? nickname : "Unbekannt")); - c = cbm_open((BYTE)4, (BYTE)4, (BYTE)0, NULL); - if (c != 0) { - c128_perror(c, "cbm_open(printer)"); - save_items(); - save_credits(); - exit(1); - } - c = cbm_write((BYTE)4, print_buffer, strlen(print_buffer)); - if (c != strlen(print_buffer)) { - c128_perror(c, "write(printer)"); - save_items(); - save_credits(); - exit(1); - } - cbm_close((BYTE)4); - log_file(print_buffer); +/* dialog which is called for each bought item */ +static signed int buy(char *name, unsigned int price) { + BYTE matches = 0; + BYTE c, nickname_len; + int einheiten; + char nickname[NICKNAME_MAX_LEN + 1]; + char rest[10]; + struct credits_t *credit; + + memset(nickname, '\0', sizeof(nickname)); + memset(rest, '\0', sizeof(rest)); + + clrscr(); + cprintf("Wieviel Einheiten \"%s\"? [1] \r\n", name); + + einheiten = cget_number(1); + + if (einheiten > 100 || einheiten < -100 || einheiten == 0) { + cprintf("\r\nEinheit nicht in [-100, 100] oder 0, Abbruch, dr" uUML "cke " + "RETURN...\r\n"); + cgetc(); + return 1; + } + + cprintf("\r\nAuf ein Guthaben kaufen? Wenn ja, Nickname eingeben:\r\n"); + { + BYTE i; + BYTE x; + BYTE y; + BYTE matches; + char *uniquematch; + input_terminator_t terminator; + while (1) { + terminator = get_input_terminated_by(INPUT_TERMINATOR_RETURN | + INPUT_TERMINATOR_SPACE, + nickname, sizeof(nickname)); + + /* Clear the screen from any previous completions */ + x = wherex(); + y = wherey(); + for (i = 1; i < 7; i++) { + /* "Completion:" is longer than NICKNAME_MAX_LEN */ + cclearxy(0, y + i, strlen("Completion:")); + } + gotoxy(x, y); + + if (terminator != INPUT_TERMINATOR_SPACE) { + break; + } + + matches = 0; + uniquematch = NULL; + for (i = 0; i < credits.num_items; i++) { + if (strncmp(nickname, credits.credits[i].nickname, strlen(nickname)) != + 0) { + continue; + } + matches++; + if (matches > 1) { + break; + } + uniquematch = credits.credits[i].nickname; + } + if (matches == 1) { + /* Display the rest of the nickname */ + textcolor(TC_LIGHT_GREEN); + cprintf("%s", uniquematch + strlen(nickname)); + textcolor(TC_LIGHT_GRAY); + strcat(nickname, uniquematch + strlen(nickname)); + } else { + /* Multiple nicknames match what was entered so far. Abort and + * display all matches, then prompt the user again. */ + char completion[NICKNAME_MAX_LEN + 1]; + BYTE len = strlen(nickname); + x = wherex(); + y = wherey(); + cprintf("\r\nCompletion:\r\n"); + matches = 0; + for (i = 0; i < credits.num_items; i++) { + if (strncmp(nickname, credits.credits[i].nickname, len) != 0) { + continue; + } + if (++matches == 5) { + cprintf("...\r\n"); + break; + } + strcpy(completion, credits.credits[i].nickname); + *(completion + len) = '\0'; + cprintf("%s", completion); + textcolor(TC_LIGHT_GREEN); + cprintf("%c", *(credits.credits[i].nickname + len)); + textcolor(TC_LIGHT_GRAY); + cprintf("%s\r\n", completion + len + 1); + } + gotoxy(x, y); + } + } + } + + if (*nickname != '\0' && *nickname != PETSCII_SP) { + nickname_len = strlen(nickname); + /* go through credits and remove the amount of money or set nickname + * to NULL if no such credit could be found */ + credit = find_credit(nickname); + if (credit != NULL) { + while ((signed int)credit->credit < ((signed int)price * einheiten)) { + if (format_euro(rest, sizeof(rest), credit->credit) == NULL) { + cprintf("Preis %d konnte nicht umgerechnet werden\r\n", + credit->credit); + exit(1); + } + cprintf( + "\r\n%s hat nicht genug Geld (%s). e) einzahlen a) abbruch \r\n", + nickname, rest); + c = cgetc(); + if (c == 'e') { + deposit_credit(nickname); + } else { + return 0; + } + } + /* substract money */ + credit->credit -= (price * einheiten); + + if (format_euro(rest, sizeof(rest), credit->credit) == NULL) { + cprintf("Preis %d konnte nicht umgerechnet werden\r\n", credit->credit); + exit(1); + } + + textcolor(TC_LIGHT_GREEN); + cprintf("\r\nVerbleibendes Guthaben f" uUML "r %s: %s. Dr" uUML + "cke RETURN...\r\n", + nickname, rest); + textcolor(TC_LIGHT_GRAY); + cget_return(); + matches++; + } else { + textcolor(TC_LIGHT_RED); + cprintf("\r\nNickname nicht gefunden in der Guthabenverwaltung! Abbruch, " + "dr" uUML "cke RETURN...\r\n"); + textcolor(TC_LIGHT_GRAY); + cget_return(); + return 0; + } + } else { + /* Ensure that nickname is NULL if it's empty because it's used in print_log + */ + *nickname = '\0'; + } + + money += price * einheiten; + items_sold += einheiten; + if (printing == 1) + print_log(name, price, einheiten, nickname, rest); + + return einheiten; } -/* Dialog, der einen durch's Abrechnen der Einträge führt */ -void buy(BYTE n) { - int negative = 1; - char entered[5] = {'1', 0, 0, 0, 0}; - BYTE i = 0, matches = 0; - BYTE c, nickname_len; - int einheiten; - char *nickname; - - if (status.status[n].item_name == NULL) { - cprintf("FEHLER: Diese Einheit existiert nicht.\r\n"); - return; - } - cprintf("Wieviel Einheiten \"%s\"? [1] \r\n", status.status[n].item_name); - while (1) { - c = getchar(); - if (c == 13) - break; - else if (c == 27) { - cprintf("Kauf abgebrochen, druecke RETURN...\r\n"); - get_input(); - return; - } else if (c == '-' && i == 0) - negative = -1; - else if (c > 47 && c < 58) - entered[i++] = c; - } - einheiten = atoi(entered) * negative; - cprintf("\r\nAuf ein Guthaben kaufen? Wenn ja, Nickname eingeben:\r\n"); - nickname = get_input(); - if (nickname != NULL && *nickname != '\0' && *nickname != 32) { - nickname_len = strlen(nickname); - /* go through credits and remove the amount of money or set nickname - * to NULL if no such credit could be found */ - for (c = 0; c < credits.num_items; ++c) - if (strncmp(nickname, credits.credits[c].nickname, nickname_len) == 0) { - if (credits.credits[c].credit < (status.status[n].price * einheiten)) { - cprintf("Sorry, %s hat nicht genug Geld :-(\r\n", nickname); - return; - } - /* Geld abziehen */ - credits.credits[c].credit -= (status.status[n].price * einheiten); - cprintf("\r\nVerbleibendes Guthaben fuer %s: %d Cents. Druecke RETURN...\r\n", - nickname, credits.credits[c].credit); - get_input(); - matches++; - break; - } - if (matches == 0) { - cprintf("\r\nNickname nicht gefunden in der Guthabenverwaltung! Abbruch, druecke RETURN...\r\n"); - get_input(); - return; - } - } else { - /* Ensure that nickname is NULL if it's empty because it's used in print_log */ - nickname = NULL; - } - status.status[n].times_sold += einheiten; - money += status.status[n].price * einheiten; - items_sold += einheiten; - if (printing == 1) - print_log(n, einheiten, nickname); +void buy_stock(BYTE n) { + if (n >= status.num_items || status.status[n].item_name == NULL) { + cprintf("FEHLER: Diese Einheit existiert nicht.\r\n"); + cget_return(); + return; + } + + status.status[n].times_sold += + buy(status.status[n].item_name, status.status[n].price); } -void set_time_interactive() { - BYTE part[3] = {'0', '0', '\0'}; - BYTE tp1, tp2, tp3; - char *time_input, *time; - cprintf("Gib die aktuelle Uhrzeit ein (Format HHMMSS):\r\n"); - time_input = get_input(); - part[0] = time_input[0]; - part[1] = time_input[1]; - tp1 = atoi(part); - part[0] = time_input[2]; - part[1] = time_input[3]; - tp2 = atoi(part); - part[0] = time_input[4]; - part[1] = time_input[5]; - tp3 = atoi(part); - set_time(tp1, tp2, tp3); - - time = get_time(); - cprintf("Zeit gesetzt: %s\r\n", time); +void buy_custom(void) { + char name[MAX_ITEM_NAME_LENGTH + 1]; + int price; + + clrscr(); + cprintf("\r\nWas soll gekauft werden?\r\n"); + if (cgetn_input(name, sizeof(name)) == 0) + return; + + cprintf("\r\nWie teuer ist \"%s\" (in cents)?\r\n", name); + + price = cget_number(0); + + if (price == 0) { + cprintf("Kauf abgebrochen, dr" uUML "cke RETURN...\r\n"); + cget_return(); + return; + } + + buy(name, price); } -int main() { - char *c; - toggle_videomode(); - /* Zeit erstmalig setzen */ - set_time_interactive(); - POKE(216, 255); - /* Variablen zurechtbiegen */ - credits.num_items = 0; - status.num_items = 0; - /* Konfigurationsdatei laden */ - load_config(); - /* Einträge (=Getränke) und Zustand laden */ - load_items(); - /* Guthaben laden */ - load_credits(); - while (1) { - /* Bildschirm anzeigen */ - print_screen(); - /* Tastatureingaben abfragen */ - c = get_input(); - /* und eventuell weitere Dialoge anzeigen */ - if (*c > 47 && *c < 58) - buy((*c) - 48); - else if (*c == 's') { - /* Zustandsdatei schreiben */ - save_items(); - save_credits(); - cprintf("Statefile/Creditfile gesichert, druecke RETURN...\r\n"); - get_input(); - } else if (*c == 'd') { - /* Drucken an- oder ausschalten */ - printing = (printing == 1 ? 0 : 1); - cprintf("Drucken ist nun %s, druecke RETURN...\r\n", - (printing == 1 ? "eingeschaltet" : "ausgeschaltet")); - get_input(); - } else if (*c == 'g') { - /* Guthabenverwalter aufrufen */ - credit_manager(); - } else if (*c == 'z') { - /* Zeit setzen */ - set_time_interactive(); - } else if (*c == 'n') { - strcpy(status.status[status.num_items].item_name, "mate"); - status.status[status.num_items].price = 23; - status.status[status.num_items].times_sold = 5; - status.num_items++; - } else if (*c == 'q') - break; - } - cprintf("BYEBYE\r\n"); +void set_time_interactive(void) { + BYTE part[3] = {'0', '0', '\0'}; + BYTE tp1, tp2, tp3; + char *time_input, *time; + cprintf("Gib die aktuelle Uhrzeit ein (Format HHMMSS):\r\n"); + time_input = get_input(); + part[0] = time_input[0]; + part[1] = time_input[1]; + tp1 = atoi(part); + part[0] = time_input[2]; + part[1] = time_input[3]; + tp2 = atoi(part); + part[0] = time_input[4]; + part[1] = time_input[5]; + tp3 = atoi(part); + set_time(tp1, tp2, tp3); + + time = get_time(); + cprintf("\r\nZeit gesetzt: %s\r\n", time); +} + +int main(void) { + char *c; + char *time; + + init_globals(); + + videomode(VIDEOMODE_80x25); + + /* clock CPU at double the speed (a whopping 2 Mhz!) */ + fast(); + + /* Manipulate the VDC with IRQs turned off. + * KERNALs default IRQ handler will also try to read the VDC status + * register, which could interfere with our code trying to read it. + */ + SEI(); + vdc_patch_charset(); + CLI(); + + clrscr(); + + /* Allocate logging buffer memory */ + init_log(); + + /* Set time initially, c128 doesn't know it */ + set_time_interactive(); + + /* disable interrupt driven VIC screen editor */ + POKE(0xD8, 255); + + /* Load configuration */ + load_config(); + + /* Load items (= drinks) */ + load_items(); + /* Load credits */ + load_credits(); + + time = get_time(); + sprintf(print_buffer, "%c----------------------------------------------------" + "----------------------------\r", + 17); + print_the_buffer(); + sprintf(print_buffer, "%cC128-Kasse Version " GV "\r", 17); + print_the_buffer(); + + sprintf(print_buffer, + "%cKasse gestartet um %s. Nutze logfile log-%u, offset %d.\r", 17, + time, log_num, log_heap_offset); + print_the_buffer(); + + print_header(); + + while (1) { + print_screen(); + c = get_input(); + /* ...display dialogs eventually */ + if (*c >= PETSCII_0 && *c <= PETSCII_9) { + /* if the input starts with a digit, we will interpret it as a number + * for the item to be sold */ + buy_stock(atoi(c)); + } else if (*c == 'f') { + buy_custom(); + } else if (*c == 's') { + cprintf("\r\nsaving items.. "); + save_items(); + cprintf("ok\r\nsaving credits.. "); + save_credits(); + cprintf("ok\r\nflushing log.. "); + log_flush(); + cprintf("ok\r\nStatefile/Creditfile/Log gesichert, dr" uUML + "cke RETURN...\r\n"); + cget_return(); + } else if (*c == 'g') { + credit_manager(); + } else if (*c == 'z') { + set_time_interactive(); + } else if (*c == 'q') + break; + } + clrscr(); + cprintf("\r\nBYEBYE\r\n"); + + return 0; }