aboutsummaryrefslogtreecommitdiff
path: root/badroff.c
blob: adf3e410607397562d6933d4077b64cbc6084588 (plain)
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
#include <stdio.h>
#include "buf.h"
#include <string.h>
#include <stdlib.h>

typedef enum {
    false,
    true
} bool;

FILE* getfile(int argc, char** argv){
    if (argc >= 2)
        return fopen(argv[1], "r");
    else
        return stdin;
}

// file read & buffer fill helpers

FILE* in;
buf* inbuf;

size_t fillbuf(size_t len){ // returns real len
    int c;
    size_t set;
    for (set = buflen(inbuf); set < len && (c = fgetc(in)) != EOF;set++)
        inschrbuf(inbuf, c);
    return set;
}

size_t chrfill(char chr){
    size_t line;
    size_t len = buflen(inbuf);
    char* str = peekstrbuf(inbuf, 0, len);
    for (line = 0; line < len; line++)
        if (str[line] == chr) return line+1;
    for (int c = 0; c != chr && (c = fgetc(in)) != EOF; line++)
        inschrbuf(inbuf, c);
    return line;
}

// typesetting config and commands to change.
int width = 80;
bool brk = false; // not implemented

char* center(char* txt){
    size_t len = strlen(txt);
    int max = width/2 + len/2; // excluding terminator
    int min = max-len+1; // first index of *txt
    char* str = malloc(sizeof(char)*(max + 1));
    str[max] = 0;
    for (int i=0; i < min; i++) str[i] = ' ';
    memcpy(str+min, txt, len);
    return str;
}
char* setwidth(char* txt){
    sscanf(txt, "%d", &width);
    return NULL;
}

char* cmds[] = {"CT ", "W "};
char* (*call[])(char* txt) = {center, setwidth};

char* cmd(void){
    char* dat = popstrbuf(inbuf, chrfill('\n'));
    size_t low = 0; size_t high = sizeof(cmds)/sizeof(*cmds); //len
    while (high >= low){
        int mid = ((unsigned int)low + (unsigned int)high) >> 1;
        char* mval = cmds[mid];

        int cmp = strncmp(dat, mval, strlen(mval));
        if (cmp < 0) // dat < mval
            high = mid - 1;
        else if (cmp > 0)
            low = mid + 1;
        else return call[mid](dat+strlen(mval));
    }
    return NULL;
}

// normal typesetting 

char* typeset(void){
    return popstrbuf(inbuf, chrfill('\n'));
}

// a parser to choose when to typeset and when to run a command

char* line(void){
    size_t sz;
    if ( (sz = fillbuf(2)) == 0)
        return "";
    char* twobytes = peekstrbuf(inbuf, 0, 2); // .., .\n, or ^.?
    if (sz == 1 || twobytes[1] == '\n') return typeset();
    if (twobytes[0] == '.') popchrbuf(inbuf);
    if (twobytes[0] == '.' && twobytes[1] != '.'){
        char* data = cmd();
        if (data) return data;
        return line();
    } else {
        return typeset();
    }
}

// orchestration

int main(int argc, char** argv){
    int c;
    in = getfile(argc, argv);
    inbuf = newbuf(256);
    if (in == NULL){
        perror(argv[1]);
        return 1;
    }
    char* out;
    while ( (out = line())[0] != '\0'){
        printf("%s",out);
    }
    fclose(in);
    return 0;
}