стирание вывода терминала в linux - PullRequest
7 голосов
/ 05 февраля 2010

Я писал программу командной строки, которая будет иметь строку состояния, очень похожую на wget.

Основная проблема, с которой я сталкиваюсь: как мне удалить то, что я уже отправил в stdout / stderr?

У меня была идея: использовать символ возврата на клавишу «\ b» и стереть результат, который я отправил. Это лучший способ? Это единственный способ? Есть ли лучший способ?

PS: я не хочу использовать что-то вроде ncurses. Обычный старый C, пожалуйста.

Спасибо


EDIT:

Могу ли я также пойти вверх и / или вниз? Пример: у меня есть 10 строк вывода, я хочу изменить третью строку с Doing ABC на ABC: Done. Как я могу это сделать?

Кроме того, кто-нибудь может опубликовать более подробную информацию о том, что символы VT102? Каковы его возможности? Пожалуйста, оставьте хорошие ссылки на это, если у вас есть какие-либо.

Спасибо

Ответы [ 7 ]

6 голосов
/ 05 февраля 2010

Основными управляющими символами форматирования являются клавиша возврата (\ b), табуляция (\ t), перевод строки (\ n) и возврат каретки (\ r). Если вам нужно больше, вы можете использовать ANSI X3.64 / ISO / IEC 6429 / ECMA-48 escape-последовательности; по крайней мере подмножество VT100 распознается большинством современных терминалов и эмуляторов. Преимущество использования ncurses заключается в том, что он будет искать возможности вашего конкретного терминала, и поэтому он будет работать, даже если ваш терминал использует другой набор escape-последовательностей.

5 голосов
/ 05 февраля 2010

Вы должны помнить, что, что касается обычных процедур stdio, stdout - это просто поток байтов без внутренних характеристик отображения; это зависит от целевого устройства, которое может быть чем угодно - от обычного терминала в стиле VT100 до печатного терминала, листового принтера, плоттера и т.д.

IMO, вам далеко лучше использовать библиотеку наподобие ncurses, чем пытаться взломать ваш собственный код управления отображением с помощью escape-кодов VT100, даже для относительно простой задачи, такой как эта Я знаю, что вы хотите придерживаться «простого старого C», но эта задача выходит за пределы простого старого C.

3 голосов
/ 05 февраля 2010

Используйте '\ r', чтобы вернуться к началу строки и, возможно, переписать всю строку.

Ищите последовательности управления VT102 - это последовательности символов ESC ... для управления вашим терминалом.

2 голосов
/ 05 февраля 2010

Существует также возможность использования Ncurses , который является библиотекой для текстового интерфейса пользователя, где этот тип поведения должен иметь некоторую поддержку. Однако это может быть излишним для чего-то подобного.

1 голос
/ 05 февраля 2010

Небольшое отклонение от вашего собственного решения:

Вы также можете распечатать возврат каретки (\r), который вернет вас в начало строки.

0 голосов
/ 05 февраля 2010

О прогресс-баре: что-то вроде этого?

#include <stdio.h>
#include <unistd.h>

typedef enum
{
    false=0,
    true=!false
} bool;

typedef struct
{
    /* Start delimiter (e.g. [ )*/
    char StartDelimiter;
    /* End Delimiter (e.g. ] )*/
    char EndDelimiter;
    /* Central block (e.g. = )*/
    char Block;
    /* Last block (e.g. > ) */
    char CurBlock;
    /* Width of the progress bar (in characters) */
    unsigned int Width;
    /* Maximum value of the progress bar */
    double Max;
    /* True if we have to print also the percentage of the operation */
    bool PrintPercentage;
    /* True if the bar must be redrawn;
       note that this must be just set to false before the first call, the function then will change it by itself.  */
    bool Update;
} ProgressBarSettings;

/* Prints/updates the progress bar */
void PrintProgressBar(double Pos, ProgressBarSettings * Settings);
/* Inits the settings of the progress bar to the default values */
void DefaultProgressBar(ProgressBarSettings * Settings);

int main()
{
    int i;
    /* Init the bar settings */
    ProgressBarSettings pbs;
    DefaultProgressBar(&pbs);
    pbs.Max=200;
    pbs.Width=60;
    printf("Progress: ");
    /* Show the empty bar */
    PrintProgressBar(0,&pbs);
    for(i=0;i<=pbs.Max;i++)
    {
        /* Wait 50 msec */
        usleep(50000);
        /* Update the progress bar */
        PrintProgressBar(i,&pbs);
    }
    puts(" Done");
    return 0;
}

/* Inits the settings of the progress bar to the default values */
void DefaultProgressBar(ProgressBarSettings * Settings)
{
    Settings->StartDelimiter='[';
    Settings->EndDelimiter=']';
    Settings->Block='=';
    Settings->CurBlock='>';
    Settings->PrintPercentage=true;
    Settings->Update=false;
    Settings->Max=100;
    Settings->Width=40;
}

/* Prints/updates the progress bar */
void PrintProgressBar(double Pos, ProgressBarSettings * Settings)
{
    /* Blocks to print */
    unsigned int printBlocks=(unsigned int)(Settings->Width*Pos/Settings->Max);
    /* Counter */
    unsigned int counter;
    /* If we are updating an existing bar...*/
    if(Settings->Update)
    {
        /* ... we get back to its first character to rewrite it... */
        for(counter=Settings->Width+2+(Settings->PrintPercentage?5:0);counter;counter--)
            putchar('\b');
    }
    else
        Settings->Update=true; /* next time we'll be updating it */
    /* Print the first delimiter */
    putchar(Settings->StartDelimiter);
    /* Reset the counter */
    counter=Settings->Width;
    /* Print all the blocks except the last; in the meantime, we decrement the counter, so in the end we'll have
       the number of spaces to fill the bar */ 
    for(;printBlocks>1;printBlocks--,counter--)
        putchar(Settings->Block);
    /* Print the last block; if the operation ended, use the normal block, otherwise the one for the last block */
    putchar((Settings->Max==Pos)?Settings->Block:Settings->CurBlock);
    /* Another block was printed, decrement the counter */ 
    counter--;
    /* Fill the rest of the bar with spaces */
    for(;counter;counter--)
        putchar(' ');
    /* Print the end delimiter */
    putchar(Settings->EndDelimiter);
    /* If asked, print also the percentage */
    if(Settings->PrintPercentage)
        printf(" %3d%%",(int)(100*Pos/Settings->Max));
    /* Flush the output buffer */
    fflush(stdout);
};

Примечание: unistd.h и usleep предназначены только для фальсификации хода операции, сам штрих-код выполнения просто использует стандартную библиотеку. Единственное предположение о выходном потоке заключается в том, что \ b фактически получает предыдущий записанный символ. Я попробовал это успешно в Windows и Linux (с gnome-терминал), не знаю, работает ли он правильно с некоторыми эмуляторами терминала. Извините за чрезмерное количество комментариев, я написал это для другого форума, где мне нужно было объяснять практически каждую строку кода новичку в Си.

0 голосов
/ 05 февраля 2010

Это индикатор выполнения для bash.

function gauge()
{
        progress="$1"
        total="$2"
        width=`tput cols`
        let gwidth=width-7

        if [ "$total" == "0" ]; then
                percent=100
        else
                set +e
                let percent=progress*100/total;
                set -e
        fi

        set +e
        let fillcount=percent*gwidth/100
        let nofillcount=gwidth-fillcount
        set -e

        fill="";
        if [ "$fillcount" -gt "0" ]; then
                for i in `seq $fillcount`; do
                        fill="$fill""|"
                done
        fi;
        nofill=""
        if [ "$nofillcount" -gt "0" ]; then
                for i in `seq $nofillcount`; do
                        nofill="$nofill"" ";
                done
        fi
        echo -e -n "\r[""$fill""$nofill""] ""$percent""%";
}
...