#include <locale.h>
#include <stdio.h>
#include <ctype.h>
#include <signal.h>
#define MAXENT 50
struct skeleton {
char prompt[20];
char keylet[5];
} bibskel[MAXENT] = {
" Author:", "%A",
" Title:", "%T",
" Journal:", "%J",
" Volume:", "%V",
" Pages:", "%P",
"Publisher:", "%I",
" City:", "%C",
" Date:", "%D",
" Other:", "%O",
" Keywords:", "%K", };
int entries = 10;
int abstract = 1;
static void addbib(FILE *, char *);
void bibedit(FILE *, char *, char *);
static void instruct(void);
static void rd_skel(char *);
static void trim(char []);
static void
usage(void)
{
puts(gettext("Usage: addbib [-p promptfile] [-a] database\n\
\t-p: the promptfile defines alternate fields\n\
\t-a: don't include prompting for the abstract\n"));
exit(1);
}
int
main(int argc, char *argv[])
{
FILE *fp, *fopen();
int i;
(void) setlocale(LC_ALL, "");
#if !defined(TEXT_DOMAIN)
#define TEXT_DOMAIN "SYS_TEST"
#endif
(void) textdomain(TEXT_DOMAIN);
if (argc == 1) {
puts(gettext(
"You must specify a bibliography file (database)."));
usage();
}
for (i = 1; argv[i][0] == '-'; i++) {
if (argv[i][1] == 'p') {
if (i >= argc - 2) {
puts(gettext("Not enough arguments for "
"-p option."));
usage();
}
rd_skel(argv[++i]);
} else if (argv[i][1] == 'a') {
if (i >= argc - 1) {
puts(gettext(
"No bibliofile specified after -a."));
usage();
}
abstract = 0;
} else {
printf(gettext(
"Invalid command line flag: %s\n"), argv[i]);
usage();
}
}
if (i < argc - 1) {
puts(gettext("Too many arguments with no options."));
usage();
}
if ((fp = fopen(argv[i], "a")) == NULL) {
perror(argv[i]);
exit(1);
}
addbib(fp, argv[i]);
return (0);
}
static void
addbib(FILE *fp, char *argv)
{
char line[BUFSIZ];
int i = 0, firstln, repeat = 0, escape = 0;
printf(gettext("Instructions? "));
fgets(line, BUFSIZ, stdin);
if (line[0] == 'y' || line[0] == 'Y')
instruct();
while (1) {
putchar('\n');
putc('\n', fp);
for (i = 0; i < entries; i++) {
printf("%s\t", gettext(bibskel[i].prompt));
if (fgets(line, BUFSIZ, stdin) == NULL) {
clearerr(stdin);
break;
}
if (line[0] == '-' && line[1] == '\n') {
i -= 2;
if (i < -1) {
printf(gettext("Too far back.\n"));
i++;
}
continue;
} else if (line[strlen(line)-2] == '\\') {
if (line[0] != '\\') {
line[strlen(line)-2] = '\n';
line[strlen(line)-1] = '\0';
trim(line);
fprintf(fp, "%s %s",
bibskel[i].keylet, line);
}
printf("> ");
again:
fgets(line, BUFSIZ, stdin);
if (line[strlen(line)-2] == '\\') {
line[strlen(line)-2] = '\n';
line[strlen(line)-1] = '\0';
trim(line);
fputs(line, fp);
printf("> ");
goto again;
}
trim(line);
fputs(line, fp);
} else if (line[0] != '\n') {
trim(line);
fprintf(fp, "%s %s", bibskel[i].keylet, line);
}
}
if (abstract) {
puts(gettext(" Abstract: (ctrl-d to end)"));
firstln = 1;
while (fgets(line, BUFSIZ, stdin)) {
if (firstln && line[0] != '%') {
fprintf(fp, "%%X ");
firstln = 0;
}
fputs(line, fp);
}
clearerr(stdin);
}
fflush(fp);
if (ferror(fp)) {
perror(argv);
exit(1);
}
editloop:
printf(gettext("\nContinue? "));
fgets(line, BUFSIZ, stdin);
if (line[0] == 'e' || line[0] == 'v') {
bibedit(fp, line, argv);
goto editloop;
}
if (line[0] == 'q' || line[0] == 'n')
return;
}
}
static void
trim(char line[])
{
int n;
n = strlen(line);
while (--n >= 0) {
if (!isspace(line[n]))
break;
}
line[++n] = '\n';
line[++n] = '\0';
}
void
bibedit(FILE *fp, char *cmd, char *arg)
{
int i = 0, status;
fclose(fp);
while (!isspace(cmd[i]))
i++;
cmd[i] = '\0';
if (fork() == 0) {
if (cmd[0] == 'v' && cmd[1] == 'i')
execlp(cmd, cmd, "+$", arg, NULL);
else
execlp(cmd, cmd, arg, NULL);
}
signal(SIGINT, SIG_IGN);
signal(SIGQUIT, SIG_IGN);
wait(&status);
signal(SIGINT, SIG_DFL);
signal(SIGQUIT, SIG_DFL);
if ((fp = fopen(arg, "a")) == NULL) {
perror(arg);
exit(1);
}
}
static void
instruct(void)
{
putchar('\n');
puts(gettext(
"Addbib will prompt you for various bibliographic fields.\n"
"If you don't need a particular field, just hit RETURN,\n"
"\tand that field will not appear in the output file.\n"
"If you want to return to previous fields in the skeleton,\n"
"\ta single minus sign will go back a field at a time.\n"
"\t(This is the best way to input multiple authors.)\n"
"If you have to continue a field or add an unusual field,\n"
"\ta trailing backslash will allow a temporary escape.\n"
"Finally, (without -a) you will be prompted for an abstract\n"
"Type in as many lines as you need, and end with a ctrl-d.\n"
"To quit, type `q' or `n' when asked if you want to continue.\n"
"To edit the database, type `edit', `vi', or `ex' instead."));
}
static void
rd_skel(char *arg)
{
FILE *pfp, *fopen();
char str[BUFSIZ];
int entry, i, j;
if ((pfp = fopen(arg, "r")) == NULL) {
fprintf(stderr, gettext("Promptfile "));
perror(arg);
exit(1);
}
for (entry = 0; fgets(str, BUFSIZ, pfp); entry++) {
for (i = 0; str[i] != '\t' && str[i] != '\n'; i++)
bibskel[entry].prompt[i] = str[i];
bibskel[entry].prompt[i] = '\0';
if (str[i] == '\n') {
fprintf(stderr, gettext(
"No tabs between promptfile fields.\n"));
fprintf(stderr, gettext(
"Format: prompt-string <TAB> %%key\n"));
exit(1);
}
for (i++, j = 0; str[i] != '\n'; i++, j++)
bibskel[entry].keylet[j] = str[i];
bibskel[entry].keylet[j] = '\0';
if (entry >= MAXENT) {
fprintf(stderr, gettext(
"Too many entries in promptfile.\n"));
exit(1);
}
}
entries = entry;
fclose(pfp);
}