💾 Archived View for gemini.spam.works › mirrors › textfiles › programming › berens.lst captured on 2022-04-29 at 00:27:29.

View Raw

More Information

⬅️ Previous capture (2020-10-31)

-=-=-=-=-=-=-

C PROCEDURE TABLES
by Tim Berens

[LISTING ONE]


/****************************************************************************
    Name : prompter.c

    Description : A routine for prompting a user for a series of answers.

#include<stdio.h>
#include"prompter.h"

struct group_stack group_stack[GROUP_STACK_SIZE];

prompter(pc)
    struct prcontrol * pc;
{
    int errstat;

    pc->current_question = 0;
    pc->group_stack_ptr = 0;

    for(;;){
        pc->errstat = 0;

        display_current_question(pc);

        gets(pc->response);

        if(*pc->response == 0){
            continue;
        }

        if(!(pc->errstat = 
          (*pc->current_group[pc->current_question].validate)(pc))){

            if(pc->errstat = 
              (*pc->current_group[pc->current_question].doit)(pc)){
                 if(pc->errstat == EXIT_NOW){
                    return(0);
                 }
            }
        }

        if(pc->current_group[pc->current_question].response != NULL){
            strcpy(pc->current_group[pc->current_question].response,
                   pc->response);
        }

        (*pc->current_group[pc->current_question].set)(pc);

        if(pc->current_group[pc->current_question].text == NULL){
            return(0);
        }

        if(pc->errstat){
            handle_error(pc->errstat,pc->errormess);
        }

    }

}

display_current_question(pc)    
    struct prcontrol * pc;
{
    printf("\n%s\n",pc->current_group[pc->current_question].text);
    printf("--->");

}

handle_error(errstat,errormess)
    int errstat;
    struct errormess * errormess;
{
    int i;
    int emess_offset = -1;
    char * message,messagebuff[100];
    
    for(i = 0 ; errormess[i].errstat != -1 ; ++i){
        if(errormess[i].errstat == errstat){
           emess_offset = i;
            break;
        }
    }
    message = messagebuff;
    if(emess_offset != -1){
        strcpy(message,errormess[emess_offset].message);
        if(errormess[emess_offset].build){
            (*errormess[emess_offset].build)(message);
        }
    }
    else{
        sprintf(message,"Error %d.",errstat);
    }

    puts("\n");
    puts(message);
    return(0);
}


/***************************************    
    Flow control routines

no_op()
{
    return(0);
}

next_question(pc)
    struct prcontrol * pc;
{
    ++pc->current_question;
    return(0);
}


pop_group(pc)
    struct prcontrol * pc;
{
    --pc->group_stack_ptr;
    pc->current_group = group_stack[pc->group_stack_ptr].group; 
    pc->current_question = group_stack[pc->group_stack_ptr].current_question;
    return(0);
}

push_current_group(pc)
    struct prcontrol * pc;
{
    group_stack[pc->group_stack_ptr].group = pc->current_group;
    group_stack[pc->group_stack_ptr].current_question = pc->current_question;
    ++pc->group_stack_ptr;
    return(0);
}

start_group(newgroup,pc)
    struct question * newgroup;
    struct prcontrol * pc;
{
    push_current_group(pc);
    pc->current_group = newgroup;
    pc->current_question = 0;
    return(0);
}

restart_group(pc)
    struct prcontrol * pc;
{
    pc->current_question = 0;
    return(0);
}


end_group(pc)   
    struct prcontrol * pc;
{
    pop_group(pc);
    ++pc->current_question;
    return(0);
}

checkerror_end_group(pc)
    struct prcontrol * pc;
{   
    if(pc->errstat){    
        return(0);
    }
    end_group(pc);
    return(0);
}

checkerror_next_question(pc)
    struct prcontrol * pc;
{   
    if(pc->errstat){    
        return(0);
    }
    next_question(pc);
    return(0);
}


[LISTING TWO]

/****************************************************************************
    Name : prompter.h 

    Description : Declarations for prompter


struct prcontrol {
    int current_question;
    struct question * current_group;
    int group_stack_ptr;
    char response[121];
    int errstat;
    struct errormess * errormess;
    };

struct question {
    char * text;
    char * response;
    int (*validate)();
    int (*doit)();
    int (*set)();
    };

struct group_stack {
    struct question * group;
    int current_question;
    };

/************************
   errormess data structure

    
struct errormess {
    int errstat;
    char * message;
    int (*build)();
    };

#define GROUP_STACK_SIZE            50
#define NO_ERROR                    0
#define EXIT_NOW                    2001

int pop_group(),end_group(),no_op(),next_question();
int checkerror_end_group(),checkerror_next_question();


[LISTING THREE]

/****************************************************************************
    Name : prsample.c

    Description : A sample that uses the prompter() routine

#include<stdio.h>
#include"prompter.h"
#include<ctype.h>

/**************************
    The report parameter variables 

char report_destination[2];
char dest_filename[30];
char single_or_range[2];
char start_account[20],end_account[20];
int account_number;
char display_parmname[50];
char include_overshort[2];

/*********************
    Error Values

#define ENTER_S_OR_R            1
#define ENTER_Y_OR_N            2
#define START_ACCOUNT_LARGER    3
#define BAD_PARM_NAME           4
#define BAD_ACCOUNT_NUMBER      5
#define ENTER_P_S_OR_D          6
#define FILE_EXISTS             7

/************************
  Report to printer, screen or disk routines

int filename_val();
struct question report_filename[] = {   
    { "What is the name of the disk file?",
        dest_filename,filename_val,no_op,checkerror_end_group},
    {   NULL,NULL,NULL,NULL,NULL }
    };

filename_val(pc)    
    struct prcontrol * pc;
{
    FILE * fp,*fopen();
    /* you should put a routine to validate that the response
       entered is a legal file name here */
    if(fp = fopen(pc->response,"r")){
        fclose(fp);
        return(FILE_EXISTS);
    }
    return(0);
}

reportdest_val(pc)
    struct prcontrol * pc;
{
    char * strchr();
    if((!strchr("PpSsDd",pc->response[0])) || (strlen(pc->response) != 1)){
        return(ENTER_P_S_OR_D);
    }
    return(0);
}

reportdest_set(pc)
    struct prcontrol * pc;
{
    char destination;
    destination = islower(*pc->response) ? *pc->response-32 : *pc->response;
    switch(destination){    
        case 'P' :
        case 'S' : next_question(pc);
                   break;
        case 'D' : start_group(report_filename,pc);
                   break;
   }
   return(0);
}

/***************************
    Account routines

int account_val(),end_account_set(),end_account_val();

struct question account_range[] = {
    {"Enter the starting account.",
        start_account,account_val,no_op,checkerror_next_question},
    {"Enter the ending account.",
        end_account,end_account_val,no_op,end_account_set},
    { NULL,NULL,NULL,NULL,NULL }
    };

int save_account_doit(),account_set();
struct question account[] = {
    {"Enter the account.",
        start_account,account_val,save_account_doit,checkerror_end_group},
    {NULL,NULL,NULL,NULL,NULL}};

account_or_range_val(pc)
    struct prcontrol * pc;
{
    char * strchr();
    if((!strchr("SsRr",pc->response[0])) || (strlen(pc->response) > 1)){
        return(ENTER_S_OR_R);
    }
    return(0);
}

account_or_range_set(pc)
    struct prcontrol * pc;
{
    char account_or_range;
    account_or_range = islower(*pc->response) ? *pc->response-32 : 
                        *pc->response;
    if(pc->errstat){
        return(0);
    }    
    if(account_or_range == 'S'){
        start_group(account,pc);
    }
    if(account_or_range == 'R'){
        start_group(account_range,pc);
    }
    return(0);
}

save_account_doit(pc)
    struct prcontrol * pc;
{
    account_number = atoi(pc->response);
    return(0);
}

account_val(pc)
    struct prcontrol * pc;
{
    if((atoi(pc->response) < 100) || (atoi(pc->response) > 1000)){   
        return(BAD_ACCOUNT_NUMBER);
    }
    return(0);
}

end_account_val(pc)
    struct prcontrol * pc;
{
    int errstat;
    if(errstat = account_val(pc)){
        return(errstat);
    }
    if(atoi(start_account) >= atoi(pc->response)){
        return(START_ACCOUNT_LARGER);
    }
    return(0);
}

end_account_set(pc)
    struct prcontrol * pc;
{
    switch(pc->errstat){
        case NO_ERROR             : end_group(pc);
                                    break;
        case START_ACCOUNT_LARGER : restart_group(pc);
                                    break;
        case BAD_ACCOUNT_NUMBER   : break;
    }
    return(0);
}

/***************************
    Get display parameters routines

char * legal_parmnames[] = {      /* In a "real" system, this table       */
    "default",                    /* would probably be stored in a file   */
    "daily",                      /* and parmname_val would check to see  */
    "weekly",                     /* if the name entered is in this file. */
    "yearly",
    NULL
    };

parmname_val(pc)
    struct prcontrol * pc;
{
    int i;
    for(i = 0 ; legal_parmnames[i] != NULL ; ++i){
        if(strcmp(pc->response,legal_parmnames[i]) == 0){
            return(0);
        }
    }    
    return(BAD_PARM_NAME);
}

bld_bad_parmname(message)
    char * message;
{
    sprintf(message + strlen(message)," %s, %s, %s, or %s.",
            legal_parmnames[0],legal_parmnames[1],legal_parmnames[2],
            legal_parmnames[3]);
    return(0);
}

/**************************
    yesno validation

yesno_val(pc)
    struct prcontrol * pc;
{
    char * strchr();
    if((!strchr("YyNn",pc->response[0])) || (strlen(pc->response) != 1)){
        return(ENTER_Y_OR_N);
    }
    return(0);
}

/**************************************
    Main question array procedure table

struct question account_parms[] = {
    {"Do you want this report for a single account or a range of accounts? (S or R)",
        single_or_range,account_or_range_val,no_op,account_or_range_set },
    {"Enter the name of the display parameter record.",
        display_parmname,parmname_val,no_op,checkerror_next_question},
    {"Do you want to include the Over/Short Report? (Y/N)",
        NULL,yesno_val,no_op,checkerror_next_question},
    {"Do you want this report on the printer, screen, or saved to disk?(P,S or D)",
        report_destination,reportdest_val,no_op,reportdest_set},
    { NULL,NULL,NULL,NULL,NULL }
    };

struct errormess account_errormess[] = { 
    { ENTER_S_OR_R,"Please enter S or R.",NULL },
    { ENTER_Y_OR_N,"Please enter Y or N.",NULL },
    { START_ACCOUNT_LARGER,"The starting account must be smaller than the ending account.",NULL },
    { BAD_ACCOUNT_NUMBER,"The account number must be between 100 and 1000",NULL },
    { BAD_PARM_NAME,"Choose one of the following :",bld_bad_parmname },
    { ENTER_P_S_OR_D,"Please enter P, S or D",NULL },
    { FILE_EXISTS,"That file already exists.",NULL },
    { -1,NULL,NULL }
    };

main(argc,argv)
    int argc;
    char * argv[];
{
    int errstat;
    struct prcontrol prcontrol;

    prcontrol.current_group = account_parms;
    prcontrol.errormess = account_errormess;

    if(errstat = prompter(&prcontrol)){
        handle_error(errstat,account_errormess);
    }
    /*  Print the report with the gathered parameters */
}




[EXAMPLE 1]


		loop{
			display current_question->text
			
			get response from user

			execute current_question->validate

			if(no error on validate){
				execute current_question->doit
			}

			copy response to current_question->response

			execute current_question->set

			if(error from validate){
				call error handler
			}
		}