После формирования escape-последовательности с помощью tparm
, ее нужно отправить на терминал. Для корректной обработки этой последовательности не следует пересылать строку на терминал с помощью функции printf
. Вместо нее примените одну из специальных функций, обеспечивающих корректную обработку любых задержек, необходимых для завершения операции, выполняемой терминалом. К ним относятся следующие:
#include
int putp(char *const str);
int tputs(char *const str, int affcnt, int (*putfunc)(int));
В случае успешного завершения функция putp
вернет константу OK
,в противном случае — ERR
. Эта функция принимает управляющую строку терминала и посылает ее в стандартный вывод stdout.
Итак, для перемещения в строку 5 и столбец 30 на экране можно применить блок программного кода, подобный приведенному далее:
char *cursor;
char *esc_sequence;
cursor = tigetstr("cup");
esc_sequence = tparm(cursor, 5, 30);
putp(esc_sequence);
Функция tputs
предназначена для ситуаций, в которых терминал не доступен через стандартный вывод stdout
, и позволяет задать функцию, применяемую для вывода символов. Она возвращает результат заданной пользователем функции putfunc
. Параметр affcnt
предназначен для обозначения количества строк, подвергшихся изменению. Обычно он устанавливается равным 1. Функция, используемая для вывода строки, должна иметь те же параметры и возвращать тип значения как у функции putfunc
. В действительности putp(string)
эквивалентна вызову tputs (string, 1, putchar)
. В следующем примере вы увидите применение функции tputs
, используемой с функцией вывода, определенной пользователем.
Имейте в виду, что в некоторых старых дистрибутивах Linux последний параметр функции tputs
определен как int (*putfunc)(char)
, что заставит вас изменить определение функции char_to_terminal
из упражнения 5.6.
Примечание
Если вы обратитесь к страницам интерактивного справочного руководства за информацией о функции tparm
и характеристиках терминалов, то можете встретить функцию tgoto
. Причина, по которой мы не используем эту функцию, хотя она, очевидно, предлагает более легкий способ перемещения курсора, заключается в том, что она не включена в стандарт X/Open (Single UNIX Specification Version 2) по данным издания 1997 г. Следовательно, мы не рекомендуем применять любую из этих функций в ваших новых программах.
Вы почти готовы добавить обработку экрана в вашу функцию выбора пункта меню. Единственно, что осталось, — очистить экран просто с помощью свойства clear
. Некоторые терминалы не поддерживают характеристику clear
, которая помещает курсор в левый верхний угол экрана. В этом случае вы можете поместить курсор в левый верхний угол и применить команду ed
— удалить до конца экрана.
Для того чтобы собрать всю полученную информацию вместе, напишем окончательную версию примера программы выбора пункта меню screenmenu.c, в которой вы "нарисуете" варианты пунктов меню на экране для того, чтобы пользователь выбрал нужный пункт (упражнение 5.6).
Упражнение 5.6. Полное управление терминалом
Вы можете переписать функцию getchoice
из программы menu4.c для предоставления полного управления терминалом. В этом листинге функция main
пропущена, потому что она не меняется. Другие отличия от программы menu4.c выделены цветом.
#include
#include
#include
#include
#include
#include
static FILE* output_stream = (FILE *)0;
char *menu[] = {
"a — add new record",
"d — delete record",
"q - quit",
NULL,
};
int getchoice(char *greet, char *choices[], FILE *in, FILE *out);
int char_to_terminal(int_char_to_write);
int main() {
...
}
int getchoice(char *greet, char* choices[], FILE[]* in, FILE* out) {
int chosen = 0;
int selected;
int screenrow, screencol = 10;
char **option;
char* cursor, *clear;
output_stream = out;
setupterm(NULL, fileno(out), (int*)0);
cursor = tigetstr("cup");
clear = tigetstr("clear");
screenrow =4;
tputs(clear, 1, (int*)char_to_terminal);
tputs(tparm(cursor, screenrow, screencol), 1, char_to_terminal);
fprintf(out, "Choice: %s", greet);
screenrow += 2;
option = choices;
while (*option) {
ftputs(tparm(cursor, screenrow, screencol), 1, char_to_terminal);
fprintf(out, "%s", *option);
screenrow++;
option++
}
Читать дальше