QT 使用QPixmap自定义光标 缩放图像模糊问题

QT中定义光标可以使用 Qt::CursorShape 预定义 的光标,也可以使用 QBitmap 和 QPixmap 自己绘制光标。QBitmap 只有黑白2色,QPixmap可以绘制彩色光标。使用QPixmap绘制光标时会出现光标边界模糊的情况,尤其是显示器设置缩放比例时,光标更是模糊。

出现模糊的原因有2个:

  1. QPixmap 投射到屏幕时自动进行了缩放,出现了图像缩放失真;
  2. 缩放时默认采用的算法导致图像信息丢失;

解决办法:

  1. 获取屏幕的缩放比例。

    复制代码
     // 获取主屏幕的缩放比例
     qreal screenRatio = QGuiApplication::primaryScreen()->devicePixelRatio();
     qreal width = 32;
     qreal height = 32;
     // 创建位图,根据屏幕缩放比例调整图形大小
     QPixmap pixmap(width*screenRatio, height*screenRatio);
     pixmap.fill(Qt::transparent);
     pixmap.setDevicePixelRatio(screenRatio);
  2. 选择更平滑的图像缩放算法。

    复制代码
     QPainter painter(&pixmap);
     painter.setRenderHint(QPainter::Antialiasing);
     // 出现缩放时选择更平滑的缩放算法 
     painter.setRenderHint(QPainter::SmoothPixmapTransform);

完整的自定义光标代码:

复制代码
/**
 * @brief 绘制光标
 * @param width 光标宽带
 * @param height 光标高度
 */
void AdjustPoint::createCursor(qreal angleOffset)
{
    qreal width = 32;
    qreal height = 32;
    // 创建位图,根据屏幕缩放比例调整图形大小
    QPixmap pixmap(width*screenRatio, height*screenRatio);
    pixmap.fill(Qt::transparent);
    pixmap.setDevicePixelRatio(screenRatio);
    // 绘制
    QPainter painter(&pixmap);
    painter.setRenderHint(QPainter::Antialiasing);
    painter.setRenderHint(QPainter::SmoothPixmapTransform);
    painter.setRenderHint(QPainter::VerticalSubpixelPositioning);
    painter.setRenderHint(QPainter::LosslessImageRendering);
    qreal winWidth = 20;
    qreal winHeight = 8;
    painter.setViewport((width - winWidth) / 2,(height - winHeight) / 2,
                        winWidth,winHeight);
    painter.setWindow(-winWidth / 2,-winHeight / 2,winWidth,winHeight);
    // 旋转光标图像到指定位置
    painter.rotate(cursorAngle + angleOffset);
    QPen pen(Qt::NoPen);
    painter.setPen(pen);
    QBrush brush(Qt::SolidPattern);
    brush.setColor(QColor(borderColor));
    painter.setBrush(brush);
    painter.drawPath(createArrowPath(winHeight+2, winWidth+1));

    brush.setColor(QColor(cursorColor));
    painter.setBrush(brush);
    painter.drawPath(createArrowPath(winHeight, winWidth));

    painter.end();

    // 创建光标,并设置
    QCursor customCursor(pixmap);
    setCursor(customCursor);
}

QPainterPath AdjustPoint::createArrowPath(qreal winHeight, qreal winWidth)
{
    QPainterPath path;
    qreal x1 = -winWidth / 2;
    qreal x2 = x1 + winHeight / 2;
    qreal x3 = -x2;
    qreal x4 = -x1;
    qreal y1 = -winHeight / 2;
    qreal y2 = -winHeight * 1 /8;
    qreal y3 = -y2;
    qreal y4 = -y1;
    path.moveTo(x1, 0);
    path.lineTo(x2,y1);
    path.lineTo(x2, y2);
    path.lineTo(x3, y2);
    path.lineTo(x3,y1);
    path.lineTo(x4, 0);
    path.lineTo(x3, y4);
    path.lineTo(x3, y3);
    path.lineTo(x2, y3);
    path.lineTo(x2, y4);
    path.lineTo(x1, 0);

    path.closeSubpath();

    return path;
}
相关推荐
神仙别闹14 小时前
基于QT(C++)实现学本科教务系统(URP系统)
数据库·c++·qt
谁动了我的代码?16 小时前
QT<34> 利用线程池处理耗时任务以及回调函数的使用
开发语言·qt
木心爱编程18 小时前
【Qt 5.14.2 新手实战】QTC++入门筑基——按钮与标签联动:QPushButton + QLabel 实现图片切换器
java·c++·qt
小新11018 小时前
vs2022+Qt插件初体验,创建带 UI 界面的 Qt 项目
开发语言·qt·ui
十五年专注C++开发19 小时前
Qt实现多语言原理和实践详解
开发语言·c++·qt·多语言
神仙别闹20 小时前
基于QT(C++)实现的翻金币游戏
c++·qt·游戏
月上林梢21 小时前
QT圆形加载进度条
数据库·c++·qt·进度条
A0_張張1 天前
记录一个PDF盖章工具(PyQt5 + PyMuPDF)
开发语言·python·qt·pdf
淼淼7631 天前
工厂方法模式
开发语言·c++·windows·qt·工厂方法模式
钱彬 (Qian Bin)1 天前
项目实践11—全球证件智能识别系统(切换为PostgreSQL数据库)
人工智能·qt·fastapi