sbase/fold.c
FRIGN eee98ed3a4 Fix coding style
It was about damn time. Consistency is very important in such a
big codebase.
2014-11-13 18:08:43 +00:00

115 lines
1.8 KiB
C

/* See LICENSE file for copyright and license details. */
#include <ctype.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include "text.h"
#include "util.h"
static void fold(FILE *, long);
static void foldline(const char *, long);
static bool bflag = false;
static bool sflag = false;
static void
usage(void)
{
eprintf("usage: %s [-bs] [-w width] [FILE...]\n", argv0);
}
int
main(int argc, char *argv[])
{
long width = 80;
FILE *fp;
ARGBEGIN {
case 'b':
bflag = true;
break;
case 's':
sflag = true;
break;
case 'w':
width = estrtol(EARGF(usage()), 0);
break;
ARGNUM:
width = ARGNUMF(0);
break;
default:
usage();
} ARGEND;
if (argc == 0) {
fold(stdin, width);
} else {
for (; argc > 0; argc--, argv++) {
if (!(fp = fopen(argv[0], "r"))) {
weprintf("fopen %s:", argv[0]);
continue;
}
fold(fp, width);
fclose(fp);
}
}
return 0;
}
static void
fold(FILE *fp, long width)
{
char *buf = NULL;
size_t size = 0;
while (agetline(&buf, &size, fp) != -1)
foldline(buf, width);
free(buf);
}
static void
foldline(const char *str, long width)
{
bool space;
long col, j;
size_t i = 0, n = 0;
int c;
do {
space = false;
for (j = i, col = 0; str[j] && col <= width; j++) {
c = str[j];
if (!UTF8_POINT(c) && !bflag)
continue;
if (sflag && isspace(c)) {
space = true;
n = j+1;
}
else if (!space)
n = j;
if (!bflag && iscntrl(c))
switch(c) {
case '\b':
col--;
break;
case '\r':
col = 0;
break;
case '\t':
col += (col+1) % 8;
break;
}
else
col++;
}
if (fwrite(&str[i], 1, n-i, stdout) != n-i)
eprintf("<stdout>: write error:");
if (str[n])
putchar('\n');
} while (str[i = n] && str[i] != '\n');
}