Qt5开发及实例V2.0-第七章-Qt图形视图框架

这篇具有很好参考价值的文章主要介绍了Qt5开发及实例V2.0-第七章-Qt图形视图框架。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

第7章 Qt 5图形视图框架

7.1 图形视图体系结构

7.1.1 Graphics View的特点

Graphics View框架结构的主要特点如下。
(1)Graphics View框架结构中,系统可以利用Qt绘图系统的反锯齿、OpenGL工具来改善绘图性能。
(2)Graphics View支持事件传播体系结构,可以使图元在场景(scene)中的交互能力提高1倍,图元能够处理键盘事件和鼠标事件。其中,鼠标事件包括鼠标按下、移动、释放和双击,还可以跟踪鼠标的移动。
(3)在Graphics View框架中,通过二元空间划分树(Binary Space Partitioning,BSP)提供快速的图元查找,这样就能够实时地显示包含上百万个图元的大场景。

7.1.2 Graphics View的三元素

它们三者之间的关系如图7.1所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

1.场景类:QGraphicsScene类
场景类主要完成的工作包括提供对它包含的图元的操作接口和传递事件、管理各个图元的状态(如选择和焦点处理)、提供无变换的绘制功能(如打印)等。
事件传播体系结构将场景事件发送给图元,同时也管理图元之间的事件传播。如果场景接收到了在某一点的鼠标单击事件,场景会将事件传给这一点的图元。
管理各个图元的状态(如选择和焦点处理)。可以通过QGraphicsScene:: setSelectionArea()函数选择图元,选择区域可以是任意的形状,使用QPainterPath表示。若要得到当前选择的图元列表,则可以使用函数QGraphicsScene:: selectedItems()。可以通过QGraphicsScene:: setFocusItem()函数或QGraphicsScene:: setFocus()函数来设置图元的焦点,获得当前具有焦点的图元使用函数QGraphicsScene::focusItem()。

2.视图类:QGraphicsView类
QGraphicsView是可滚动的窗口部件,可以提供滚动条来浏览大的场景。如果需要使用OpenGL,则可以使用QGraphicsView::setViewport()将视图设置为QGLWidget。
视图接收键盘和鼠标的输入事件,并将它们翻译为场景事件(将坐标转换为场景的坐标)。使用变换矩阵函数QGraphicsView::matrix()可以变换场景的坐标,实现场景缩放和旋转。QGraphicsView提供QGraphicsView::mapToScene()和QGraphicsView:: mapFromScene()用于与场景的坐标进行转换。

3.图元类:QGraphicsItem类
它是场景中各个图元的基类,在它的基础上可以继承出各种图元类,Qt已经预置的包括直线(QGraphicsLineItem)、椭圆(QGraphicsEllipseItem)、文本图元(QGraphicsTextItem)、矩形(QGraphicsRectItem)等。
QGraphicsItem主要有以下功能。
 处理鼠标按下、移动、释放、双击、悬停、滚轮和右键菜单事件。
 处理键盘输入事件。
 处理拖曳事件。
 分组。
 碰撞检测。

7.1.3 GraphicsView的坐标系统

1.场景坐标
场景坐标是所有图元的基础坐标系统。场景坐标系统描述了顶层的图元,每个图元都有场景坐标和相应的包容框。场景坐标的原点在场景中心,坐标原点是x轴正方向向右,y轴正方向向下。
QGraphicsScene类的坐标系以中心为原点(0,0),如图7.2所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

2.视图坐标
视图坐标是窗口部件的坐标。视图坐标的单位是像素。QGraphicsView视图的左上角是(0,0),x轴正方向向右,y轴正方向向下。所有的鼠标事件最开始都是使用视图坐标。
QGraphicsView类继承自QWidget类,因此它与其他的QWidget类一样,以窗口的左上角作为自己坐标系的原点,如图7.3所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

3.图元坐标
图元使用自己的本地坐标,这个坐标系统通常以图元中心为原点,这也是所有变换的原点。图元坐标方向是x轴正方向向右,y轴正方向向下。创建图元后,只需注意图元坐标就可以了,QGraphicsScene和QGraphicsView会完成所有的变换。
QGraphicsItem类的坐标系,若在调用QGraphicsItem类的paint()函数重绘图元时,则以此坐标系为基准,如图7.4所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux
Graphics View框架提供了多种坐标变换函数,见表7.1。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

7.2 【实例】:图形视图

7.2.1 飞舞的蝴蝶

【例】(难度中等)(CH701)设计界面,一个蝴蝶在屏幕上不停地上下飞舞。
操作步骤如下。
(1)新建Qt Widgets Application(详见1.3.1节),项目名为“Butterfly”,基类选择“QMainWindow”,类名命名默认为“MainWindow”,取消“创建界面”复选框的选中状态。单击“下一步”按钮,最后单击“完成”按钮,完成该项目工程的建立。
(2)在“Butterfly”项目名上单击鼠标右键,在弹出的快捷菜单中选择“添加新文件…”菜单项,在弹出的对话框中选择“C++ Class”选项。单击“Choose…”按钮,弹出对话框,在“Base class”后面的下拉列表框中选择基类名“QObject”,在“Class name”后面的文本框中输入类的名称“Butterfly”。
(3)单击“下一步”按钮,单击“完成”按钮,添加文件“butterfly.h”和“butterfly. cpp”。

(4)Butterfly类继承自QObject类、QGraphicsItem类,在头文件“butterfly.h”中完成的代码具体内容。
(5)在源文件“butterfly. cpp”中完成的代码具体内容如下:

#include "butterfly.h"
#include <math.h>
const static double PI=3.1416;
Butterfly::Butterfly(QObject *parent) : QObject(parent)
{
    up = true;			//给标志蝴蝶翅膀位置的变量赋初值
    pix_up.load("up.png");		//调用QPixmap的load()函数加载所用到的图片
    pix_down.load("down.png");
    startTimer(100);		//启动定时器,并设置时间间隔为100毫秒
}

boundingRect()函数为图元限定区域范围。此范围是以图元自身的坐标系为基础设定的。具体实现代码内容如下:

QRectF Butterfly::boundingRect() const
{
    qreal adjust =2;
    return QRectF(-pix_up.width()/2-adjust,-pix_up.height()/2-adjust,
                  pix_up.width()+adjust*2,pix_up.height()+adjust*2);
}

在重画函数paint()中,首先判断当前已显示的图片是pix_up还是pix_down。实现蝴蝶翅膀上下飞舞效果时,若当前显示的是pix_up图片,则重绘pix_down图片,反之亦然。具体实现代码内容如下:

void Butterfly::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
{
    if(up)
    {
        painter->drawPixmap(boundingRect().topLeft(),pix_up);
        up=!up;
    }
    else
    {
        painter->drawPixmap(boundingRect().topLeft(),pix_down);
        up=!up;
    }
}

定时器的timerEvent()函数实现蝴蝶的飞舞,具体实现代码内容如下:

void Butterfly::timerEvent(QTimerEvent *)
{
    //边界控制
    qreal edgex=scene()->sceneRect().right()+boundingRect().width()/2;
					//限定蝴蝶飞舞的右边界
    qreal edgetop=scene()->sceneRect().top()+boundingRect(). height()/2;							//限定蝴蝶飞舞的上边界
    qreal edgebottom=scene()->sceneRect().bottom()+boundingRect(). height()/2;				//限定蝴蝶飞舞的下边界
    if(pos().x()>=edgex)			//若超过了右边界,则水平移回左边界处
        setPos(scene()->sceneRect().left(),pos().y());
    if(pos().y()<=edgetop)			//若超过了上边界,则垂直移回下边界处
        setPos(pos().x(),scene()->sceneRect().bottom());
    if(pos().y()>=edgebottom)		//若超过了下边界,则垂直移回上边界处
        setPos(pos().x(),scene()->sceneRect().top());
    angle+=(qrand()%10)/20.0;
    qreal dx=fabs(sin(angle*PI)*10.0);
    qreal dy=(qrand()%20)-10.0;
    setPos(mapToParent(dx,dy));		//(a)
}

(6)完成了蝴蝶图元的实现后,在源文件“main.cpp”中将它加载到场景中,并关联一个视图,具体实现代码内容如下:

#include <QApplication>
#include "butterfly.h"
#include <QGraphicsScene>
int main(int argc,char* argv[])
{
    QApplication a(argc,argv);
    QGraphicsScene *scene = new QGraphicsScene;
    scene->setSceneRect(QRectF(-200,-200,400,400));
    Butterfly *butterfly = new Butterfly;
    butterfly->setPos(-100,0);
    scene->addItem(butterfly);
    QGraphicsView *view = new QGraphicsView;
    view->setScene(scene);
    view->resize(400,400);
    view->show();
    return a.exec();
}

(7)运行程序,将程序中用到的图片保存到该工程的D:\Qt\CH7\CH701\build-Butterfly-Desktop_Qt_5_8_0_MinGW_32bit-Debug文件夹中,运行结果如图7.5所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

7.2.2 地图浏览器

【例】(难度中等)(CH702)设计一个地图浏览器,包括地图的浏览、放大、缩小,以及显示各点的坐标等,如图7.6所示。

操作步骤如下。
(1)新建Qt Widgets Application (详见1.3.1节),项目名称为“MapWidget”,基类选择“QMainWindow”,类名命名默认为“MainWindow”,取消“创建界面”复选框的选中状态。单击“下一步”按钮,最后单击“完成”按钮,完成该项目工程的建立。
(2)在“MapWidget”项目名上单击鼠标右键,在弹出的快捷菜单中选择“添加新文件…”菜单项,在弹出的对话框中选择“C++ Class”选项。单击“Choose…”按钮,弹出对话框,在“Base class”后面的文本框中输入基类名“QGraphicsView”(手工添加),在“Class name”后面的文本框中输入类的名称“MapWidget”。
(3)单击“下一步”按钮,单击“完成”按钮,添加文件“mapwidget.h”和文件“mapwidget.cpp”。
(4)MapWidget类继承自QGraphicsView类,作为地图浏览器的主窗体。在头文件“mapwidget.h”中完成的代码。
(5)在源文件“mapwidget.cpp”中完成的代码。
(6)新建一个文本文件“maps.txt”,利用该文本文件描述与地图相关的信息,将该文件保存在该工程下的D:\Qt\CH7\CH702\ build-MapWidget-Desktop_Qt_5_ 4_0_ MinGW_32bit-Debug文件中,文件内容为:
China.jpg 114.4665527 35.96022297 119.9597168 31.3911575
(7)打开“mapwidget.cpp”文件,添加读取地图信息readMap()函数的具体实现代码如下:

void MapWidget::readMap()            			//读取地图信息
{
    QString mapName;
    QFile mapFile("maps.txt");			//(a)
    int ok = mapFile.open(QIODevice::ReadOnly);//以“只读”方式打开此文件
    if(ok)								//分别读取地图的名称和四个经纬度信息
    {
        QTextStream ts(&mapFile);
        if(!ts.atEnd())
        {
            ts>>mapName;
            ts>>x1>>y1>>x2>>y2;
        }
    }
    map.load(mapName);							//将地图读取至私有变量map中
}

根据缩放滑动条的当前值,确定缩放的比例,调用scale()函数实现地图缩放。完成地图缩放功能的slotZoom()函数的具体实现代码内容如下:

void MapWidget::slotZoom(int value)   	//地图缩放
{
    qreal s;
    if(value>zoom)                 		//放大
    {
        s=pow(1.01,(value-zoom));
    }
    else                             		//缩小
    {
        s=pow(1/1.01,(zoom-value));
    }
    scale(s,s);
    zoom = value;
}

QGraphicsView类的drawBackground()函数中以地图图片重绘场景的背景来实现地图显示。具体实现代码如下:

void MapWidget::drawBackground(QPainter *painter, const QRectF &rect)
{
    painter->drawPixmap(int(sceneRect().left()),int(sceneRect(). top()), map);
}

响应鼠标移动事件mouseMoveEvent()函数,完成某点在各层坐标中的映射及显示。具体实现代码如下:

void MapWidget::mouseMoveEvent(QMouseEvent *event)
{
    //QGraphicsView 坐标
    QPoint viewPoint = event->pos();
    viewCoord->setText(QString::number(viewPoint.x())+","+
                       QString::number(viewPoint.y()));
    //QGraphicsScene 坐标
    QPointF scenePoint = mapToScene(viewPoint);
    sceneCoord->setText(QString::number(scenePoint.x())+","+
                        QString::number(scenePoint.y()));
    //地图坐标(经、纬度值)
    QPointF latLon = mapToMap(scenePoint);
    mapCoord->setText(QString::number(latLon.x())+","+
                      QString::number(latLon.y()));
}

完成从场景坐标至地图坐标的转换mapToMap()函数。具体实现代码如下:

QPointF MapWidget::mapToMap(QPointF p)
{
    QPointF latLon;
    qreal w =sceneRect().width();
    qreal h =sceneRect().height();
    qreal lon = y1-((h/2+p.y())*abs(y1-y2)/h);
    qreal lat = x1+((w/2+p.x())*abs(x1-x2)/w);
    latLon.setX(lat);
    latLon.setY(lon);
    return latLon;
}

(8)下面是文件“main.cpp”的具体代码:

#include <QApplication>
#include "mapwidget.h"
#include <QFont>
int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    QFont font("ARPL KaitiM GB",12);
    font.setBold(true);
    a.setFont(font);
    MapWidget mapWidget;
    mapWidget.show();
    return a.exec();
}

(9)将程序用到的图片保存到该工程的D:\Qt\CH7\CH702\build-MapWidget-Desktop_Qt_5_8_0_MinGW_32bit-Debug文件夹中,运行结果如图7.6所示。

7.2.3 图元创建

【例】(难度中等)(CH703)设计窗体,显示各种类型QGraphicsItem(包括不停闪烁的圆及来回移动的星星等),如图7.7所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

实现步骤如下。
(1)新建Qt Widgets Application(详见1.3.1节),项目名称为“GraphicsItem”,基类选择“QMainWindow”,类名命名默认为“MainWindow”,取消“创建界面”复选框的选中状态。单击“下一步”按钮,最后单击“完成”按钮,完成该项目工程的建立。
(2)MainWindow类继承自QMainWindow作为主窗体,包含一个加入图元的各种操作的菜单栏,以及一个显示各种类型图元的QGraphicsView作为主窗体的centralWidget。“mainwindow.h”文件的具体代码实现内容。
(3)“mainwindow.cpp”文件中的代码。
(4)将程序中所用图片保存到该工程的D:\Qt\CH7\CH703\build-GraphicsItem-Desktop_Qt_5_8_0_MinGW_32bit-Debug文件夹下,此时运行效果如图7.8所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux
以上完成了主窗体的显示工作,下面介绍如何实现圆的闪烁功能。
(1)在“GraphicsItem”项目名上单击鼠标右键,在弹出的快捷菜单中选择“添加新文件…”菜单项,在弹出的对话框中选择“C++ Class”选项。单击“Choose…”按钮,弹出对话框,在“Base class”后面的下拉列表框中选择基类名“QObject”,在“Class name”后面的文本框中输入类的名称“FlashItem”。
(2)单击“下一步”按钮,单击“完成”按钮,添加文件“flashitem.h”和文件“flashitem.cpp”。

(3)“flashitem.h”文件的具体代码如下:

#include <QGraphicsItem>
#include <QPainter>
class FlashItem : public QObject,public QGraphicsItem
{
    Q_OBJECT
public:
    explicit FlashItem(QObject *parent = 0);
    QRectF boundingRect() const;
    void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget);
    void timerEvent(QTimerEvent *);
private:
    bool flash;
    QTimer *timer;
signals:
public slots:
};

(4)“flashitem.cpp”文件的具体代码如下:

#include "flashitem.h"
FlashItem::FlashItem(QObject *parent) :  QObject(parent)
{
    flash=true;			//为颜色切换标识赋初值
    setFlag(ItemIsMovable);	//(a)
    startTimer(1000);		//启动一个定时器,以1000毫秒为时间间隔
}

定义图元边界的函数boundingRect(),完成以图元坐标系为基础,增加两个像素点的冗余工作。具体实现代码如下:

QRectF FlashItem::boundingRect() const
{
    qreal adjust = 2;
    return QRectF(-10-adjust,-10-adjust,43+adjust,43+adjust);
}

为自定义图元重绘的函数paint()具体实现代码如下:

void FlashItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
{
    painter->setPen(Qt::NoPen);		//闪烁图元的阴影区不绘制边线
    painter->setBrush(Qt::darkGray);		//闪烁图元的阴影区的阴影画刷颜色为深灰
    painter->drawEllipse(-7,-7,40,40);	//绘制阴影区
    painter->setPen(QPen(Qt::black,0));	
				//闪烁区的椭圆边线颜色为黑色、线宽为0
    painter->setBrush(flash?(Qt::red):(Qt::yellow)); //(a)
    painter->drawEllipse(-10,-10,40,40);	      //(b)
}

定时器响应函数timerEvent()完成颜色切换标识的反置,并在每次反置后调用update()函数重绘图元以实现闪烁的效果。具体实现代码如下:

void FlashItem::timerEvent(QTimerEvent *)
{
    flash=!flash;
    update();
}

(5)在“mainwindow.h”文件中添加代码如下:

public slots:    
    void slotAddFlashItem();
private:
    QAction *addFlashItemAct;

(6)在“mainwindow.cpp”文件中添加代码如下:

#include "flashitem.h"

其中,在createActions()函数中添加代码如下:

addFlashItemAct = new QAction(tr("加入闪烁圆"),this);
connect(addFlashItemAct,SIGNAL(triggered()),this,SLOT(slotAddFlashItem()));

在createMenus()函数中添加代码如下:

itemsMenu->addAction(addFlashItemAct);

在initScene()函数中添加代码如下:

for(i=0;i<3;i++)      
        slotAddFlashItem();

函数slotAddFlashItem()具体实现代码如下:

void MainWindow::slotAddFlashItem()  	//在场景中加入一个闪烁图元
{
    FlashItem *item = new FlashItem;
    scene->addItem(item);
    item->setPos((qrand()%int(scene->sceneRect().width()))-200,
                 (qrand()%int(scene->sceneRect().height()))-200);
}

(7)运行效果如图7.9所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux

下面将接着完成实现星星移动的功能。
(1)向项目中添加一个新的C++类,类名命名为“StartItem”,操作步骤同前。StartItem类继承自QGraphicsItem类,实际上是一个图片图元。
“startitem.h”文件的具体代码如下:

#include <QGraphicsItem>
#include <QPainter>
class StartItem : public QGraphicsItem
{
public:
    StartItem();
    QRectF boundingRect() const;
    void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget);
private:
    QPixmap pix;
};

(2)StartItem构造函数中仅完成读取图片信息的工作。
“startitem.cpp”文件中的具体代码如下:

#include "startitem.h"
StartItem::StartItem()
{
    pix.load("star.png");
}

定义图元的边界函数boundingRect(),它是所有自定义图元均必须实现的函数,代码如下:

QRectF StartItem::boundingRect() const
{
    return QRectF(-pix.width()/2,-pix.height()/2,pix.width(),pix. height());
}

自定义图元重绘函数paint(),代码如下:

void StartItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option,QWidget *widget)
{
    painter->drawPixmap(boundingRect().topLeft(),pix);
}

(3)在“mainwindow.h”文件中添加代码如下:

public slots:    
    void slotAddAnimationItem();
private:
    QAction *addAnimItemAct;

(4)在“mainwindow.cpp”文件中添加代码如下:

#include "startitem.h"
#include <QGraphicsItemAnimation>
#include <QTimeLine>

其中,在createActions()函数中添加代码如下:

addAnimItemAct = new QAction(tr("加入 星星"),this);
connect(addAnimItemAct,SIGNAL(triggered()),this,SLOT(slotAddAnimationItem()));

在createMenus()函数中添加代码如下:
itemsMenu->addAction(addAnimItemAct);
在initScene()函数中添加代码如下:

for(i=0;i<3;i++)      
    slotAddAnimationItem();

实现函数slotAddAnimationItem()的具体代码如下:

void MainWindow::slotAddAnimationItem() 	//在场景中加入一个动画星星
{
    StartItem *item = new StartItem;
    QGraphicsItemAnimation *anim = new QGraphicsItemAnimation;
    anim->setItem(item);
    QTimeLine *timeLine = new QTimeLine(4000);
    timeLine->setCurveShape(QTimeLine::SineCurve);
    timeLine->setLoopCount(0);
    anim->setTimeLine(timeLine);
    int y =(qrand()%400)-200;
    for(int i=0;i<400;i++)
    {
        anim->setPosAt(i/400.0,QPointF(i-200,y));
    }
    timeLine->start();
    scene->addItem(item);
}

(5)最终运行结果如图7.7所示,图中的小星星会不停地左右移动。

7.2.4 图元的旋转、缩放、切变和位移

【例】(难度中等)(CH704)设计界面,实现蝴蝶各种变形。如图7.10所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux
(1)新建Qt Widgets Application(详见1.3.1节),项目名称为“ItemWidget”,基类选择“QWidget”,类名命名为“MainWidget”,取消“创建界面”复选框的选中状态。单击“下一步”按钮,最后单击“完成”按钮,完成该项目工程的建立。
(2)MainWidget类继承自QWidget,作为主窗体类,用于对图元的显示,包含一个控制面板区及一个显示区。

“mainwidget.h”文件中的代码如下:

#include <QWidget>
#include <QGraphicsView>
#include <QGraphicsScene>
#include <QFrame>
class MainWidget : public QWidget
{
    Q_OBJECT
public:
    MainWidget(QWidget *parent = 0);
    ~MainWidget();
    void createControlFrame();
private:
    int angle;
    qreal scaleValue;
    qreal shearValue;
    qreal translateValue;
    QGraphicsView *view;
    QFrame *ctrlFrame;
};

(3)“mainwidget.cpp”文件中的具体代码。
右侧的控制面板区分为旋转控制区、缩放控制区、切变控制区及位移控制区,每个区均由包含一个QSlider对象的QGroupBox对象实现,具体实现代码。
(4)运行效果如图7.11所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux
上面完成的是主窗体的功能,下面介绍用于变形显示的图元的制作。
(1)在“ItemWidget”项目名上单击鼠标右键,在弹出的快捷菜单中选择“添加新文件…”菜单项,在弹出的对话框中选择“C++ Class”选项。单击“Choose…”按钮,弹出对话框,在“Base class”后面的文本框中输入基类名“QGraphicsItem”(手工添加),在“Class name”后面的文本框中输入类的名称“PixItem”。
(2)单击“下一步”按钮,单击“完成”按钮,添加文件“pixitem.h”和文件“pixitem.cpp”。
(3)自定义PixItem类继承自QGraphicsItem类。

“pixitem.h”文件中的具体代码如下:

#include <QGraphicsItem>
#include <QPixmap>
#include <QPainter>
class PixItem : public QGraphicsItem
{
public:
    PixItem(QPixmap *pixmap);
    QRectF boundingRect() const;
    void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget);
private:
    QPixmap pix;     		//作为图元显示的图片
};

(4)PixItem的构造函数只是初始化了变量pix。“pixitem.cpp”文件中的具体内容如下:
#include “pixitem.h”

PixItem::PixItem(QPixmap *pixmap)
{
    pix = *pixmap;
}

定义图元边界的函数boundingRect(),完成以图元坐标系为基础增加两个像素点的冗余的工作。具体实现代码如下:

QRectF PixItem::boundingRect() const
{
    return QRectF(-2-pix.width()/2,-2-pix.height()/2,pix.width()+4, pix. height()+4);
}

重画函数只需QPainter的drawPixmap()函数将图元图片绘出即可。具体代码如下:

void PixItem::paint(QPainter *painter, const QStyleOptionGraphicsItem  *option,QWidget *widget)
{
    painter->drawPixmap(-pix.width()/2,-pix.height()/2,pix);
}

(5)在“mainwidget.h”文件中添加代码如下:

#include "pixitem.h"
private:
PixItem *pixItem;

(6)打开“mainwidget.cpp”文件,在语句scene->setSceneRect(-200,-200,400,400)与view = new QGraphicsView之间添加如下代码:

QPixmap *pixmap = new  QPixmap("image.png");
pixItem = new PixItem(pixmap);
scene->addItem(pixItem);
pixItem->setPos(0,0);

(7)运行效果如图7.12所示。
Qt5开发及实例V2.0-第七章-Qt图形视图框架,QT5,ARM MCU,Linux,qt,数据库,服务器,开发语言,运维,linux
上述内容只是完成了图元图片的加载显示。下面介绍实现图元的各种变形的实际功能。
(1)在“mainwidget.h”文件中添加槽函数声明如下:

public slots:
    void slotRotate(int);
    void slotScale(int);
    void slotShear(int);
    void slotTranslate(int);

(2)在“mainwidget.cpp”文件中添加头文件:

#include <math.h>

其中,在createControlFrame()函数中的QVBoxLayout *frameLayout=new QVBoxLayout语句之前添加以下代码:

connect(rotateSlider,SIGNAL(valueChanged(int)),this,SLOT(slotRotate(int)));
connect(scaleSlider,SIGNAL(valueChanged(int)),this,SLOT(slotScale(int)));
connect(shearSlider,SIGNAL(valueChanged(int)),this,SLOT(slotShear(int)));
connect(translateSlider,SIGNAL(valueChanged(int)),this,SLOT(slotTranslate(int)));

实现图元的旋转功能函数slotRotate(),主要是调用QGraphicsView类的rotate()函数实现的,它的参数为旋转角度的度数值,具体实现代码如下:

void MainWidget::slotRotate(int value)
{
    view->rotate(value-angle);
    angle = value;
}

实现图元的缩放功能函数slotScale(),主要是调用QGraphicsView类的scale()函数实现的,它的参数为缩放的比例,具体实现代码如下:

void MainWidget::slotScale(int value)
{
    qreal s;
    if(value>scaleValue)
        s=pow(1.1,(value-scaleValue));
    else
        s=pow(1/1.1,(scaleValue-value));
    view->scale(s,s);
    scaleValue=value;
}

实现图元的切变功能函数slotShear(),主要是调用QGraphicsView类的shear()函数实现的,它的参数为切变的比例,具体实现代码如下:

void MainWidget::slotShear(int value)
{
    view->shear((value-shearValue)/10.0,0);
    shearValue=value;
}

实现图元的位移功能函数slotTranslate(),主要是调用QGraphicsView类的translate()函数实现的,它的参数为位移的大小,具体实现代码如下:

void MainWidget::slotTranslate(int value)
{
    view->translate(value-translateValue,value-translateValue);
    translateValue=value;
}

(3)最终运行结果如图7.10所示,读者可以试着拖曳滑块观看图形的各种变换效果。



本章相关例程源码下载

1.Qt5开发及实例_CH701.rar 下载

Qt5开发及实例_CH701.rar

2.Qt5开发及实例_CH702.rar 下载

Qt5开发及实例_CH702.rar

3.Qt5开发及实例_CH703.rar 下载

Qt5开发及实例_CH703.rar

4.Qt5开发及实例_CH704.rar 下载

Qt5开发及实例_CH704.rar文章来源地址https://www.toymoban.com/news/detail-730993.html

到了这里,关于Qt5开发及实例V2.0-第七章-Qt图形视图框架的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 第七章:敏捷开发工具方法-part1-敏捷开发基础

    敏捷开发背景 速度是企业竞争致胜的关键因素,软件项目的最大挑战在于一方面需要应付变动中的需求,一方面需要在有限的时间完成项目,传统的软件工程难以满足这些要求 所以软件团队除了在技术上必须日益精进,更需要运用有效的开发流程,以确保团队能够发挥综效

    2024年02月09日
    浏览(66)
  • 【鸿蒙开发】第七章 ArkTS语言UI范式-基础语法

    通过前面的章节,我们基本清楚鸿蒙应用开发用到的语言和项目基本结构,在【鸿蒙开发】第四章 Stage应用模型及项目结构也提到过ArkTS的UI范式的 基本语法 、 状态管理 、 渲染控制 等能力,简要介绍如下: 基本语法 : ArkTS 定义了 声明式UI描述 、 自定义组件 和 动态扩展

    2024年02月03日
    浏览(64)
  • 第七章:敏捷开发工具方法-part2-CI/CD工具介绍

    什么是CI/Cd? CI-Continuous integration : 持续集成 是指多名开发者在开发不同功能代码的过程当中,可以频繁的将代吗行合并到一起并切相互不影响工作。 CD-continuous deployment : 持续部署 是基于某种工具或平台实现代码自动化的构建、测试和部署到线上环境以实现交付高质量的

    2024年02月09日
    浏览(56)
  • python第七章(字典)

    一。字典(类型为dict)的特点: 1.符号为大括号 2.数据为键值对形式出现 3.各个键值对之间以逗号隔开 格式:str1={\\\'name\\\':\\\'Tom\\\'}  name相当于键值(key),Tom相当于值 二。空字典的创建方法 三。字典的基本操作(增删改查) 1.字典的增加操作:字典序列[key] = 值 注意点:如果存

    2024年01月24日
    浏览(57)
  • [JavaScript] 第七章 对象

    🌹作者主页:青花锁 🌹简介:Java领域优质创作者🏆、Java微服务架构公号作者😄 🌹简历模板、学习资料、面试题库、技术互助 🌹文末获取联系方式 📝 [Java项目实战] 介绍Java组件安装、使用;手写框架等 [Aws服务器实战] Aws Linux服务器上操作nginx、git、JDK、Vue等 [Java微服务

    2024年02月02日
    浏览(132)
  • 第七章 测试

    7.1.1 选择程序设计语言 1. 计算机程序设计语言基本上可以分为汇编语言和高级语言 2. 从应用特点看,高级语言可分为基础语言、结构化语言、专用语言 01 有理想的模块化机制; 02 可读性好的控制结构和数据结构; 03 便于调试和提高软件可靠性; 04 编译程序发现程序错误的

    2024年02月08日
    浏览(79)
  • 第七章 图论

    第七章 图论 一、数据结构定义 图的邻接矩阵存储法 图的邻接表存储法 把所有节点存储为节点数组,每个节点里有自己的数据和一个边指针,这个边指针相当于一个链表的头指针,这个链表里存放所有与这个节点相连的边,边里存放该边指向的节点编号和下一条边指针 图的

    2024年02月14日
    浏览(96)
  • 数据结构第七章

    图(Graph)G由两个集合V和E组成,记为G=(V, E),其中V是顶点的有穷非空集合,E是V中顶点偶对的有穷集合,这些顶点偶对称为边。V(G)和E(G)通常分别表示图G的顶点集合和边集合,E(G)可以为空集。若EG)为空,则图G只有顶点而没有边。 子图:假设有两个图G=(V,E)和G1=(V1,E1);如果V1

    2024年02月03日
    浏览(65)
  • 第七章金融中介

             金融中介是通过向资金盈余者发行 间接融资合约( 如存款单),并和资金短缺者达成 间接投资合约 (发放信贷)或购买其发行的证券,在资金供求方之间融通资金,对资金跨期、跨域进行优化配置的金融机构。         金融体系由金融市场和金融中介构成,以银行业为

    2024年02月04日
    浏览(59)
  • 第七章 函数矩阵

    和矩阵函数不同的是,函数矩阵本质上是一个矩阵,是以函数作为元素的矩阵。 矩阵函数本质上是一个矩阵,是以矩阵作为自变量的函数。 函数矩阵和数字矩阵的运算法则完全相同。 不过矩阵的元素 a i j ( x ) a_{ij}(x) a ij ​ ( x ) 需要是闭区间 [ a , b ] [a,b] [ a , b ] 上的实函数

    2024年02月04日
    浏览(63)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包