我正在创建一个程序,在其中我使用QProcess框架在Ubuntu 16.04 Qt 5.5.1上运行进程,并启用了C++ 11。我将进程输出流定向到QTextEdit。
我想给这个输出着色,使其与本地终端使用嵌入的ANSI转义颜色序列解释的颜色相同。然而,我无法解析转义序列,因为它们似乎在QProcess输出中丢失了。我最初认为QString正在剥离它们,但经过一些测试后,我不相信这是事实。
我发现some information为我指明了ANSI转义颜色解释的方向,如果我可以在QProcess输出中保留转义序列的话。
下面是我在Qt代码中所做的一个示例项目。
源文件…
#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QString>
#include <QProcess>
#include <QStringList>
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);
QStringList input = {"gcc will_not_build.c"};
QProcess * proc = new QProcess();
proc->setReadChannel(QProcess::StandardOutput);
proc->setProcessChannelMode(QProcess::MergedChannels);
proc->setWorkingDirectory("/path/to/test/c/file/");
//Start bash
proc->start("bash");
proc->waitForStarted();
// Write as many commands to this process as needed
foreach(QString str, input){
proc->write(str.toUtf8() + "\n");
proc->waitForBytesWritten(-1);
}
// Let bash close gracefully
proc->write("exit $?\n");
proc->waitForBytesWritten(-1);
proc->closeWriteChannel();
proc->waitForFinished();
proc->waitForReadyRead();
QByteArray read_data = proc->readAll();
// The use of tr(read_data) also works here.
QString output = tr(read_data);//QString::fromStdString (read_data.toStdString ());
proc->closeReadChannel(QProcess::StandardOutput);
proc->close();
delete proc;
// Add the output to the text box
ui->textEdit->append (output);
}
MainWindow::~MainWindow()
{
delete ui;
}
头文件...
#ifndef MAINWINDOW_H
#define MAINWINDOW_H
#include <QMainWindow>
namespace Ui {
class MainWindow;
}
class MainWindow : public QMainWindow
{
Q_OBJECT
public:
explicit MainWindow(QWidget *parent = 0);
~MainWindow();
private:
Ui::MainWindow *ui;
};
#endif // MAINWINDOW_H
表单文件...
<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
<class>MainWindow</class>
<widget class="QMainWindow" name="MainWindow">
<property name="geometry">
<rect>
<x>0</x>
<y>0</y>
<width>400</width>
<height>300</height>
</rect>
</property>
<property name="windowTitle">
<string>MainWindow</string>
</property>
<widget class="QWidget" name="centralWidget">
<widget class="QTextEdit" name="textEdit">
<property name="geometry">
<rect>
<x>33</x>
<y>19</y>
<width>331</width>
<height>211</height>
</rect>
</property>
</widget>
</widget>
<widget class="QMenuBar" name="menuBar">
<property name="geometry">
<rect>
<x>0</x>
<y>0</y>
<width>400</width>
<height>19</height>
</rect>
</property>
</widget>
<widget class="QToolBar" name="mainToolBar">
<attribute name="toolBarArea">
<enum>TopToolBarArea</enum>
</attribute>
<attribute name="toolBarBreak">
<bool>false</bool>
</attribute>
</widget>
<widget class="QStatusBar" name="statusBar"/>
</widget>
<layoutdefault spacing="6" margin="11"/>
<resources/>
<connections/>
</ui>
C源文件
int main(){
// Intentionally will not build
I will not build :)
}
我的输出看起来像这样:
QProcess gcc输出
原生Linux终端的输出如下所示:
带颜色的Linux终端gcc输出
有谁知道如何在QProcess输出中保留ANSI转义颜色序列,以便模拟Linux终端颜色?
顺便说一下,我在Qt Creator的源代码中找到了一个类,它可以将ANSI转义颜色转换为Rich Text颜色,所以我知道有人已经走上了这条路。
2条答案
按热度按时间wqlqzqxt1#
QProcess
不会干扰进程输出,只是gcc
--和许多其他发出彩色输出的程序一样--默认情况下,只有当它检测到正在TTY设备上写入时,才会发出彩色转义序列。如果您想禁用这个启发式算法并要求总是生成彩色输出,则必须在编译器命令行中添加
-fdiagnostics-color=always
选项。lf5gs5x22#
感谢一个非常有见地的回答我的问题,我能够找到一个解决我的问题。我将分享...
QProcess没有问题,QString也没有问题。问题在于程序执行的环境。由于这些程序的输出 (gcc等) 没有连接到TTY设备,所有ANSI转义序列都被剥离。不过有一种方法to trick the output to appear as if it were connected to a TTY device。
只需在命令前添加
unbuffer
即可。由于我的用途实际上是创建一个Qt Creator插件,我已经链接了大部分Qt Creator源代码。碰巧已经存在一个名为
AnsiEscapeCodeHandler
的方便类,可以将ANSI转义序列转换为QTextCharFormat's和相应的ANSI转义序列剥离字符串。为了说明我如何使用这个类,但现在在我的例子中,我将从可下载的Qt Creator源代码中复制
ansieescapecodehandler.h
和ansiescapecodehandler.cpp
到我的测试项目中。我不得不从AnsiEscapeCodeHandler
源文件中删除几行,以便在Qt Creator源代码的其余部分的上下文之外编译,但仅此而已。新的源文件...
新的头文件...
新的彩色输出... QProcess gcc output