- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个宽度为 ("imagewidth") 和高度为 ("imageheight") 的 16 位图像。数据当前存储在一个长度为("imagewidth"*"imageheight")的unsigned short int数组中
我想从我称为“数据”的数据集中创建一个 16 位灰度 QImage(使用 Qt 5.14)。
这是我正在使用的代码:
QImage image = Qimage(imagewidth,imageheight,QImage::Format_Grayscale16);
for(int i=0;i<imagewidth;i++)
{
for(int j=0;j<imageheight;j++)
{
uint pixelval = data[i+j*imagewidth];
QRgb color = qRgb(pixelval, pixelval, pixelval);
image.setPixel(i,j, color);
}
}
代码正在运行,我正在获取图像,但我只能获取以 255 为增量的值...所以 0、255...
如何将每个像素的实际像素值设置为 0 到 65535?
最佳答案
QRgba64
是 16 位(每个组件)颜色的正确选择。
另一种选择是使用 QImage::pixelColor()
检索颜色(并使用 QImage::setPixelColor()
设置)应该或多或少与深度无关。
函数qRgb()
是一个糟糕的选择,因为它有意处理 8 位(每个组件)颜色。
来自 Qt 文档:
QRgb QColor::qRgb(int r, int g, int b)
Returns the ARGB quadruplet (255, r, g, b).
alpha 的值 255 给出了第一个提示,但检查结果类型 QRgb
使这一点显而易见:
typedef QColor::QRgb
An ARGB quadruplet on the format #AARRGGBB, equivalent to an unsigned int.
一览source code on woboq.org支持这个:
inline Q_DECL_CONSTEXPR QRgb qRgb(int r, int g, int b)// set RGB value
{ return (0xffu << 24) | ((r & 0xffu) << 16) | ((g & 0xffu) << 8) | (b & 0xffu); }
一个小例子来说明这一点:
#include <QtWidgets>
int main(int argc, char **argv)
{
qDebug() << "Qt Version:" << QT_VERSION_STR;
qDebug() << "qRgb(1 << 12, 1 << 12, 1 << 12):"
<< hex << qRgb(1 << 12, 1 << 12, 1 << 12);
qDebug() << "QColor(qRgb(1 << 12, 1 << 12, 1 << 12)):"
<< QColor(qRgb(1 << 12, 1 << 12, 1 << 12));
qDebug() << "QColor().fromRgba64(1 << 12, 1 << 12, 1 << 12):"
<< QColor().fromRgba64(1 << 12, 1 << 12, 1 << 12);
// done
return 0;
}
输出:
Qt Version: 5.11.2
qRgb(1 << 12, 1 << 12, 1 << 12): ff000000
QColor(qRgb(1 << 12, 1 << 12, 1 << 12)): QColor(ARGB 1, 0, 0, 0)
QColor().fromRgba64(1 << 12, 1 << 12, 1 << 12): QColor(ARGB 1, 0.062501, 0.062501, 0.062501)
使用 QColor
和同伴的替代方法是将 data
值直接写入图像:
QImage image = Qimage(imagewidth, imageheight, QImage::Format_Grayscale16);
for (int j = 0; j < imageheight; ++j) {
quint16 *dst = (quint16*)(image.bits() + j * image.bytesPerLine());
for (int i = 0; i < imagewidth; ++i) {
dst[i] = data[i + j * imagewidth];
}
}
这肯定比将 data
值转换为颜色再转换为灰度级更快更准确。
请注意,我将循环换成了行和列。处理源 (data
) 和目标 (dst
) 中的连续字节将提高缓存局部性并在速度上获得返回。
在撰写本文时,具有 16 位深度的图像在 Qt 中是相当新的。
Qt 5.12 中添加了每个组件 16 位深度的新颜色格式:
QImage::Format_RGBX64 = 25
QImage::Format_RGBA64 = 26
QImage::Format_RGBA64_Premultiplied = 27
Qt 5.13 中加入了 16 位深度的灰度:
QImage::Format_Grayscale16 = 28
(文档复制自enum QImage::Format
)
为了解决这个问题,我制作了一个示例应用程序,用于将每分量 16 位的 RGB 图像转换为 16 位灰度。
testQImageGray16.cc
:
#include <QtWidgets>
QImage imageToGray16(const QImage &qImg)
{
QImage qImgGray(qImg.width(), qImg.height(), QImage::Format_Grayscale16);
for (int y = 0; y < qImg.height(); ++y) {
for (int x = 0; x < qImg.width(); ++x) {
qImgGray.setPixelColor(x, y, qImg.pixelColor(x, y));
}
}
return qImgGray;
}
class Canvas: public QWidget {
private:
QImage _qImg;
public:
std::function<void(QPoint)> sigMouseMove;
public:
Canvas(QWidget *pQParent = nullptr):
QWidget(pQParent)
{
setMouseTracking(true);
}
Canvas(const QImage &qImg, QWidget *pQParent = nullptr):
QWidget(pQParent), _qImg(qImg)
{
setMouseTracking(true);
}
virtual ~Canvas() = default;
Canvas(const Canvas&) = delete;
Canvas& operator=(const Canvas&) = delete;
public:
virtual QSize sizeHint() const { return _qImg.size(); }
const QImage& image() const { return _qImg; }
void setImage(const QImage &qImg) { _qImg = qImg; update(); }
protected:
virtual void paintEvent(QPaintEvent *pQEvent) override;
virtual void mouseMoveEvent(QMouseEvent *pQEvent) override;
};
void Canvas::paintEvent(QPaintEvent *pQEvent)
{
QWidget::paintEvent(pQEvent);
QPainter qPainter(this);
qPainter.drawImage(0, 0, _qImg);
}
void Canvas::mouseMoveEvent(QMouseEvent *pQEvent)
{
if (sigMouseMove) sigMouseMove(pQEvent->pos());
}
QString getInfo(const QImage &qImg)
{
QString qStr;
QDebug(&qStr) << "Image Info:\n" << qImg;
for (int i = 0, len = qStr.length(); i < qStr.length(); ++i) {
if (qStr[i] == ',' && i + 1 < len && qStr[i + 1] != ' ') qStr[i] = '\n';
}
return qStr;
}
QString getPixelInfo(const QImage &qImg, QPoint pos)
{
if (!QRect(QPoint(0, 0), qImg.size()).contains(pos)) return QString();
const int bytes = (qImg.depth() + 7) / 8; assert(bytes > 0);
const QByteArray data(
(const char*)(qImg.bits()
+ pos.y() * qImg.bytesPerLine()
+ (pos.x() * qImg.depth() + 7) / 8),
bytes);
QString qStr;
QDebug(&qStr) << pos << ":" << qImg.pixelColor(pos)
<< "raw:" << QString("#") + data.toHex();
return qStr;
}
int main(int argc, char **argv)
{
qDebug() << "Qt Version:" << QT_VERSION_STR;
QApplication app(argc, argv);
// load sample data
const QImage qImgRGB16("pnggrad16rgb.png"/*, QImage::Format_RGBX64*/);
// setup GUI
QWidget winMain;
QGridLayout qGrid;
int col = 0, row = 0;
QLabel qLblRGBInfo(getInfo(qImgRGB16));
qGrid.addWidget(&qLblRGBInfo, row++, col);
Canvas qCanvasRGB(qImgRGB16);
qGrid.addWidget(&qCanvasRGB, row++, col);
QLabel qLblRGB;
qGrid.addWidget(&qLblRGB, row++, col);
row = 0; ++col;
Canvas qCanvasGray(qImgRGB16.convertToFormat(QImage::Format_Grayscale16));
QLabel qLblGrayInfo;
qGrid.addWidget(&qLblGrayInfo, row++, col);
qGrid.addWidget(&qCanvasGray, row++, col);
QLabel qLblGray;
qGrid.addWidget(&qLblGray, row++, col);
QHBoxLayout qHBoxQImageConvert;
QButtonGroup qBtnGrpQImageConvert;
QRadioButton qTglQImageConvertBuiltIn("Use QImage::convertToFormat()");
qBtnGrpQImageConvert.addButton(&qTglQImageConvertBuiltIn);
qTglQImageConvertBuiltIn.setChecked(true);
qHBoxQImageConvert.addWidget(&qTglQImageConvertBuiltIn);
QRadioButton qTglQImageConvertCustom("Use imageToGray16()");
qBtnGrpQImageConvert.addButton(&qTglQImageConvertCustom);
qHBoxQImageConvert.addWidget(&qTglQImageConvertCustom);
qGrid.addLayout(&qHBoxQImageConvert, row++, col);
winMain.setLayout(&qGrid);
winMain.show();
// install signal handlers
auto updatePixelInfo = [&](QPoint pos)
{
qLblRGB.setText (getPixelInfo(qCanvasRGB.image(), pos));
qLblGray.setText(getPixelInfo(qCanvasGray.image(), pos));
};
qCanvasRGB.sigMouseMove = updatePixelInfo;
qCanvasGray.sigMouseMove = updatePixelInfo;
auto updateGrayImage = [&](bool customConvert)
{
qCanvasGray.setImage(customConvert
? qImgRGB16.convertToFormat(QImage::Format_Grayscale16)
: imageToGray16(qImgRGB16));
qLblGrayInfo.setText(getInfo(qCanvasGray.image()));
qLblGray.setText(QString());
};
QObject::connect(&qTglQImageConvertBuiltIn, &QRadioButton::toggled,
[&](bool checked) { if (checked) updateGrayImage(false); });
QObject::connect(&qTglQImageConvertCustom, &QRadioButton::toggled,
[&](bool checked) { if (checked) updateGrayImage(true); });
// runtime loop
updateGrayImage(false);
return app.exec();
}
和 CMake CMakeLists.txt
的构建脚本:
project(QImageGray16)
cmake_minimum_required(VERSION 3.10.0)
set_property(GLOBAL PROPERTY USE_FOLDERS ON)
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED ON)
set(CMAKE_CXX_EXTENSIONS OFF)
find_package(Qt5Widgets CONFIG REQUIRED)
include_directories("${CMAKE_SOURCE_DIR}")
add_executable(testQImageGray16
testQImageGray16.cc)
target_link_libraries(testQImageGray16
Qt5::Widgets)
# define QT_NO_KEYWORDS to prevent confusion between of Qt signal-slots and
# other signal-slot APIs
target_compile_definitions(testQImageGray16 PUBLIC QT_NO_KEYWORDS)
我下载了一个示例图片 pnggrad16rgb.png
来自 www.fnordware.com/superpng/samples.html . (其他示例图像可以在 The "official" test-suite for PNG 上找到。)
在 VS2017 中构建并运行后,我得到了以下快照:
当鼠标移到显示的图像上时,底部标签显示图像中的当前位置和相应的像素作为 QColor
和原始十六进制值。
出于好奇,我用两个嵌套循环实现了 OP 的方法(修复了 qRgb()
问题):
QImage imageToGray16(const QImage &qImg)
{
QImage qImgGray(qImg.width(), qImg.height(), QImage::Format_Grayscale16);
for (int y = 0; y < qImg.height(); ++y) {
for (int x = 0; x < qImg.width(); ++x) {
qImgGray.setPixelColor(x, y, qImg.pixelColor(x, y));
}
}
return qImgGray;
}
将结果与 QImage::convertToFormat()
进行比较这是我在 Qt 文档中找到的。
QImage QImage::convertToFormat(QImage::Format format, Qt::ImageConversionFlags flags = Qt::AutoColor) const &
QImage QImage::convertToFormat(QImage::Format format, Qt::ImageConversionFlags flags = Qt::AutoColor) &&
Returns a copy of the image in the given format.
The specified image conversion flags control how the image data is handled during the conversion process.
这很有趣:
没有明显的区别。不过,考虑到显示器可能会将颜色深度降低到 8 或 10,这可能并不令人惊讶(除了人类可能无法区分 216 灰色阴影或 2 163 RGB 值。
然而,通过删除图像,我意识到 QImage::convertToFormat()
,每个像素的第一个和第二个字节始终相同(例如 b2b2
),使用 imageToGray16()
进行自定义转换时情况并非如此。
我没有深入挖掘,但值得进一步研究这些方法中哪种方法实际上更准确。
关于c++ - 设置16位灰度QImage的像素值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57061509/
为什么这个脚本不起作用?仅当页面宽度超过 915 像素时,我希望单击按钮时滚动页面 100 像素。我试图通过仅在宽度超过 915 像素时允许该函数运行来实现此目的,但它没有发生。怎么办? $(docu
我需要您帮助我建立一个网站。我是一个新手,已经得到了一个设计为 900 像素宽的主体,但他们给了我一个 1200 像素宽的图像。他们希望图像跨越整个 1200 像素的宽度,因此页面两侧基本上会有 30
我有一个在 y 轴上展开的 UIScrollview 并调用这个委托(delegate)方法: -(void)scrollViewDidScroll:(UIScrollView *)scrollVie
我有一个固定的标题这个标题在我滚动时改变高度和图像标志但是当我调整窗口大小时我希望图像保持比例但随着我缩小浏览器而变得更小标志只有在限制时缩小浏览器靠近图像,但我希望在调整浏览器大小时图像变小。 我该
在我的项目中,我使用 ArcGIS API for JavaScript https://developers.arcgis.com/javascript/但是对于(在这里插入非常大的坏词)我无法覆盖
有没有办法使用 jQuery,根据窗口滚动的距离做不同的事情? 这是我现在使用的代码; $(document).scroll(function() { // If scroll distanc
这基本上是 Jetpack Joyride 中运动的基本版本,但不是 Joyrider 以每秒 100 像素的速度下降,而是字母“x”从控制台的正中间以每秒 100 像素的速度下降和点击事件会导致它以
我像这样处理 MINMAXINFO: case WM_GETMINMAXINFO: { LPMINMAXINFO p_info = (LPMINMAXINFO)lPar
我对 javascript 有点陌生,我一直在查找 documentElement、clientWidth 和 clientHeight 并试图找出为什么它将我的 Canvas 设置为 300px x
我正在编写一些软件来读取 DICOM 文件,但我不确定如何处理具有未定义长度的标签。标准是这样说的 “如果值字段具有显式长度,则值长度字段应包含等于长度(以字节为单位)的值 值字段。否则,值字段 有一
我对 OpenGL 有点陌生,但我很确定我的问题在于所使用的像素格式,或者我的纹理是如何生成的...... 我正在使用 16 位 RGB5_A1 像素格式在平面 2D 四边形上绘制纹理,但在这个阶段我
有没有办法获取直播电视流,例如在像素级别上进行分析。 我的目标是检查直播电视流(例如使用java),例如广播电台 Logo 是否可见。 有机会通过 Google 电视观看此直播吗? 是否有机会通过笔记
我正在尝试构建一个函数,它以给定角度从特定坐标延伸,并循环遍历该线上的像素,直到遇到黑色像素. 如果角度为 180 度,这很容易实现。在这种情况下,搜索只会向下扩展,在每次迭代中将列坐标加 1。然而,
我已经研究了一段时间,但找不到任何解决方案。 这是我的代码 如果您将此代码复制并粘贴到本网站的 HTML 区域:http://jsfiddle.net/T3Nnu/3/ 如果您查看 Facebo
我有一个网页 - http://bit.ly/YHFX5B如果你看一下页脚,你会发现它后面有一些额外的白色像素/线条。我不明白他们是从哪里来的。 请告知他们可能来自哪里。 谢谢,丹 最佳答案 在 #f
如何在没有状态栏和操作栏的情况下获取屏幕高度(像素)或者如果有人告诉我如何获取状态栏和操作栏的高度,它也会有所帮助。我已经找到了屏幕高度,但它包括状态栏和操作栏.我将支持库 v7 用于操作栏。我在网上
Java 字符串根据宽度(像素)换行 在一些场景下,我们经常会通过判断字符串的长度,比如个数来实现换行,但是中文、英文、数字、其实在展示的时候同样长度的字符串,其实它的宽度是不一样的,这也是们我通
我创建了一个不错的简单可扩展列表。它应该像单选列表一样工作,您应该只能选择一个元素。我还没有实现这部分,因为我对列表的大小有疑问: class ExpandableListRadio extends
我使用以下代码滚动到元素顶部,但我想滚动到元素顶部上方 10px,不知道如何执行此操作,有什么建议吗?谢谢! $('html, body').stop(true,true).animate({
我有一个链接,可以在滚动时更改其垂直位置。当我点击此链接时,我想(平滑地)转到页面上的某个位置,该位置距离页面顶部正好 1080 像素。 我无法实现它,希望有人能帮助我。 链接: 脚本: $(do
我是一名优秀的程序员,十分优秀!