QT显示机制

了解QT显示机制,最重要的就是要了解QT是如何管理窗体的显示区域的,这里有个重要的类:QRegion, 在QT中可以通过QRegion定义一个窗体的显示区域,也可以通过QRegion定义窗体的可修改区域,比如在QPainter()中通过QPainter::setClipRect设定一个区域,我们绘图则只能在这个区域,此区域外绘图都是无效的。通过QRegion可以作一系列的逻辑运算,如两个区域相加,相减等。QRegion定义的区域不一定是连续的,但一定是由封闭的区域组成的,我们常会碰到一个窗体的显示区域被其他窗体分割为几块的情况。QT对这些显示区域的管理,类似于对窗体的管理,也是通过服务器与客户端的方式。参照以前的说法Server表示为全局的Global
,客户端为本地得Local。那么WindowsServer管理一个全局的显示区域即所有的Top-Level widget显示区域。而其他的child windget 的管理则在每一个QT应用程序中由QWSRegionManager管理,Top_Level widget 显示区域也会加载在其中,这个不难理解,因为Server只是负责将窗体事件发送到客户端,具体处理还是由客户端来操作。具体的流程还是来看代码吧。

显示区域管理者QWSRegionManager的初始化
服务器:
通过调用openDisplay()。
客户端:  
在QWSDisplayData类的构造函数中通过调用QWSDisplayData::init()完成。

考虑一个比较简单的情况,我们要显示的widget 是一个Top_Leverl widget。在调用Show()函数中,这个widget将通过showWindows()向服务器请求做三件事:(以下窗体是指在global windows statck 中的TOP_Level widget)
1:调用QWSDisplay::requestRegion向服务器请求窗体显示区域。
2:调用QWSDdisplay::setAltitude向服务器请求设置窗体的优先级。此优先级是指在windows statck中的位置,而不是指QWSWidow 中的窗体优先级属性。Windows statck 中的第一个窗体就是显示在LCD上最前面的窗体。
3:调用QWSDisplay::requestFocus 向服务器请求设置窗体为焦点窗体。焦点窗体能接收Key, Mouse 事件,但不是所有的焦点窗体都能接收Key,Mouse 事件,如果有窗体设置为GrabKey 或则GrabMouse 则Key, Mouse 事件将分别传递至此窗体。
下面将通过代码分析winddows Server对这三个请求的处理过程:
一: QWSDisplay::requestRegion的处理
void QWSServer::invokeRegion( QWSRegionCommand *cmd, QWSClient *client )
{
    ................
    QRegion region;
    region.setRects(cmd->rectangles, cmd->simpleData.nrectangles);
    if ( !region.isEmpty() )
changingw->setNeedAck( TRUE );
    bool isShow = !changingw->isVisible() && !region.isEmpty();
    setWindowRegion( changingw, region ); //***设置窗体显示区域
    syncRegions( changingw );                //***通知客户端 刷新显示区域
    if ( isShow )
emit windowEvent( changingw, Show );
    if ( !region.isEmpty() )
emit windowEvent( changingw, Geometry );
    else
emit windowEvent( changingw, Hide );
    if ( focusw == changingw && region.isEmpty() )
setFocus(changingw,FALSE);
    .................
}
invokeRegion调用setWindowRegion设置窗体显示区域,调用syncRegions通知客户端 刷新显示区域,并产生一些窗体事件如:Show, Geometry,Hide 。
setWindowRegion函数的实现如下:
QRegion QWSServer::setWindowRegion( QWSWindow* changingw, QRegion r )
{
QRegion exposed;
    if (changingw) {
changingw->requested_region = r;
r = r - serverRegion;                      //exposed不为空则有显示区域被释放
exposed = changingw->allocation() - r; //低等级窗体增加可见区域
    } else {                                     
exposed = serverRegion-r;
serverRegion = r;
    }
    QRegion extra_allocation;
int windex = -1;

bool deeper = changingw == 0;
    for (uint i=0; i<windows.count(); i++) {
QWSWindow* w = windows.at(i);
if ( w == changingw ) {
     windex = i;
     extra_allocation = r - w->allocation(); //如果extra_allocation不为空
     deeper = TRUE;                               //需要增加新的新的显示区域
} else if ( deeper ) {
     w->removeAllocation(rgnMan, r);//低优先级窗体去掉被覆盖的区域
     r -= w->allocation();//如果r为空 则更低优先级的窗体被完全覆盖
} else {                     //如果窗体是第一次调用Show 直接走这
     //higher windows
     r -= w->allocation();//如果r为空 则窗体被高优先级窗体完全覆盖
}
if ( r.isEmpty() ) {     //窗体被完全覆盖
     break; // Nothing left for deeper windows
     }
}
...................
if ( changingw && !changingw->requested_region.isEmpty() )
changingw->addAllocation( rgnMan, extra_allocation & screenRegion );
     //为changingw窗体增加新的可见区域 置modifed标志为TRUE
else if ( !disablePainting )
paintServerRegion();

    exposeRegion( exposed, windex+1 );//增加低级窗体可见区域。
    return exposed;
}
注:增加新的显示区域不一定是整个显示区域的面积增大了,而是显示区域的块变多了。 一个显示区域可能由多个不连续和连续的Region组成。

void QWSServer::exposeRegion( QRegion r, int start )
{
    r &= screenRegion;
    for (uint i=start; i<windows.count(); i++) {
if ( r.isEmpty() )                              //可见区域为空
     break; // Nothing left for deeper windows
QWSWindow* w = windows.at(i);
w->addAllocation( rgnMan, r ); //增加新的可见区域 置modifed标志为TRUE
r -= w->allocation();                           //r 更低级窗体可见区域
    }
    dirtyBackground |= r; //得到需要刷新的背景区域 如果r为空 则新增区域为0
}
exposeRegion为低等级窗体增加可见区域。

syncRegions:此函数主要是向客户端发送RegionModified事件,真正的绘图也是由客户端来完成。 还是通过代码来分析:
void QWSServer::syncRegions( QWSWindow *active )
{
    rgnMan->commit();   //拷贝数据到一段共享内存,服务器为读写权限,客户端为只读
    notifyModified( active );//通过客户端显示区域已更改,客户端绘制相关区域
    paintBackground( dirtyBackground );//绘制背景区域修改部分。
    dirtyBackground = QRegion();
}

void QWSServer::notifyModified( QWSWindow *active )
{
    // notify active window first
    if ( active )
active->updateAllocation();    //首先通知active 窗体
    // now the rest //通知所有modified标志为TRUE的窗体
    for (uint i=0; i<windows.count(); i++) {
QWSWindow* w = windows.at(i);
w->updateAllocation();     
    }
}
void QWSWindow::updateAllocation()
{
    if ( modified || needAck) {
c->sendRegionModifyEvent( id, exposed, needAck ); // 发送消息
exposed = QRegion();                                  //复位低级窗体新增显示区域
modified = FALSE; //modified为真表示窗体的显示区域被修改。
needAck = FALSE;
    }
}

 

客户端对RegionModifyEvent的处理。
客户端接收到消息后会调用translateRegionModifiedEvent函数来进行处理
bool QETWidget::translateRegionModifiedEvent( const QWSRegionModifiedEvent *event )
{
    QWSRegionManager *rgnMan = qt_fbdpy->regionManager();
    if ( alloc_region_index < 0 ) {
alloc_region_index = rgnMan->find( winId() ); //从共享内存中得到region索引
if ( alloc_region_index < 0 ) {
     return FALSE;
}
    }
    QWSDisplay::grab();
    int revision = *rgnMan->revision( alloc_region_index );
    if ( revision != alloc_region_revision ) {
alloc_region_revision = revision;
QRegion newRegion = rgnMan->region( alloc_region_index );//得到显示区域
QWSDisplay::ungrab();                                          
alloc_region = newRegion;

// set children's allocated region dirty
................
    } else {
QWSDisplay::ungrab();
    }
    if ( event->simpleData.nrectangles )
    {                                               // alloc_region >= exposed
QRegion exposed;                            //需要刷新区域的大小
exposed.setRects( event->rectangles, event->simpleData.nrectangles );
QSize s( qt_screen->deviceWidth(), qt_screen->deviceHeight() );
exposed = qt_screen->mapFromDevice( exposed, s );
qwsUpdateActivePainters();
repaintDecoration( exposed, FALSE );//绘制窗体的一些修饰如边框,caption等
repaintHierarchy( exposed, FALSE );      //绘制窗体显示区域及子窗体通过发送
    }                                               //PaintEvent事件到各窗体
    qws_regionRequest = FALSE;
    return TRUE;
}
repaintHierarchy函数中所有需要刷新的子窗体都会收到Paint事件。在Paint事件中,开始绘图。显示中只刷新exposed这个区域而不是将分配的区域alloc_region 全部刷新一次,这样做可以提高效率。

二:QWSDdisplay::setAltitude 的处理
invokeSetAltitude(const QWSChangeAltitudeCommand *cmd,
       QWSClient *client)
{
    int winId = cmd->simpleData.windowid;
    int alt = cmd->simpleData.altitude;
    bool fixed = cmd->simpleData.fixed;
    ...................
     QWSWindow* changingw = findWindow(winId, 0);
    ...................
    changingw->setNeedAck( TRUE );
    if ( fixed && alt >= 1) {
changingw->onTop = TRUE;
    }
    if ( alt < 0 )
lowerWindow( changingw, alt ); //窗体优先级下降
    else
raiseWindow( changingw, alt ); // 提升窗体优先级
    if ( !changingw->forClient(client) ) {
refresh();
    }
}
invokeSetAltitude通过调用lowerWindow,raiseWindow来调整窗体的优先级,如果一个Widget被显示,即调用Show此时alt == 0; 如果alt == 1则此窗体应该为最上层,如果alt == 2则窗体位FULL-SCREEN即全屏显示的窗体,可以通过setWFlags(WStyle_StaysOnTop) 来设定这个属性。 优先级较高的窗体将被优先显示, 在没有显式通过SetRegionPriority命令来改变窗体优先级的话,在Windows Stack中窗体将按照后进的优先级较高为原则。
可以参考 insertPrioritizedWindow函数,在qt-embedded-free-3.3.6 可能没有这个函数,因为在这个版本中不存在窗体优先级,除了WStyle_StaysOnTop属性的窗体为第一级优先级外,其他窗体都按照后进的优先为原则。

三:QWSDisplay::requestFocus 的处理请参考invokeSetFocus函数。

时间: 2024-09-20 05:59:48

QT显示机制的相关文章

qt-Qt 点击运行,QT显示“应用程序输出”,但是没有窗口显示,但任务管理器里有运行

问题描述 Qt 点击运行,QT显示"应用程序输出",但是没有窗口显示,但任务管理器里有运行 本来是可以运行的,但是我添加了几个已有项 .h和.cpp 然后在运行就出现这种情况了,我用vb来做也是这样,(原来的.h和.cpp是可以在C++的main函数调用,在小黑框正常运行的) 解决方案 窗口有show()吗? 解决方案二: 那就把那几个文件去掉,再试试看,如果想加进来,一个一个的加,慢慢排除原因. 解决方案三: 只添加.c和.h,没有修改main的话,如果没有编译错误,是不会有问题的.

Qt之显示网络图片

简述 Qt中包含了网络模块-network,我们可以很容易的进行各种网络编程和数据传输,关于network的类很多,其中包含:支持DNS.HTTP.TCP/UDP等众多高级类,可以参考助手. 下面我们先看一个简单地示例:Qt显示一个网络图片. 简述 效果 源码 处理方式 效果 源码 创建按钮及显示图像的标签,连接信号槽. m_pButton = new QPushButton(this); m_pButton->setText(QStringLiteral("开始下载")); m

【转】QT事件传递与事件过滤器

      [概览]     1.重载特定事件函数.    比如: mousePressEvent(),keyPressEvent(),  paintEvent() .     2.重新实现QObject::event().    这一般用在Qt没有提供该事件的处理函数时.也就是,我们增加新的事件时.    3.安装事件过滤器    比如用 objA 过滤 objB 的事件,即事件到达 objB 之前,先交由 objA 处理.只需两个步骤:     调用objB->installEventFil

用于显示分层数据的嵌套网格(from msdn)

数据|显示 在"前沿技术"的 2003 年 8 月刊,我讨论了如何扩展 ASP.NET DataGrid 服务器控件,以便将多表数据容器(如 DataSet 对象)用作其数据源. 如果 DataSet 包含数对相关表,则只要所显示的表是其中某个关系的父级,该控件就将添加动态创建的按钮列. 当单击此列按钮时,将显示子 DataGrid,并将根据此关系列出选定记录的子行. 总体行为显示在图 1 中,此行为与 Windows 窗体 DataGrid 控件在类似情形下的工作方式相同. 图 1

JAVA程序的类加载及其反射机制

JAVA中类文件加载是动态的.JVM指令是被封装在了. class文件里面,而.class文件的加载过程是动态的,也就是说当我们用到的时候才会去加载,如果不用的话,就不会去加载我们的类.这里所说的用到包括两种方式,第一种就是new一个对象的时候(这个时候要特别注意,当设计到多态的时候,就会有一点点变化,这时候编译器会做一些优化,这样以来当加载的时候会提前加载设计到多态的类,关于这一点下面有个例子(example 1)来说明.另一种就是当一个类的静态代码被调用的时候. java 代码 //exam

QT语音声卡调试教程

  操作教程 1.先在QT进房后的界面上点击"话筒"右边的倒三角符号 选择语音检测 2.在声音输入选择合适的声卡,并说话 如果QT显示系统成功检测到声音输入,选择下一步. 3.在声音输出选择合适的声卡,并点击播放测试音按钮对已选播放设备进行测试 如果您能听到QT播放的测试音,那么恭喜,您的扬声器是正常的,点击关闭按钮开始正常语音吧. XP系统 如果您的系统是XP系统,双击右下角系统的小喇叭,在选项那里选择属性 把所有音量控制都点上 把所有音量都静音,然后分别打开并听一下效果,也可以调节

qt-关于Qt的图片处理的问题

问题描述 关于Qt的图片处理的问题 用qt显示一张图片,图片上有三个不规则的图形A B C.想做到这样的效果,点击某个图形的时候使该图形变色. 强调一点,图形是不规则的. 请问如何能做到这一点 解决方案 背景和图片分别是单一颜色的么?如果是的话,用QRgb检测图片的像素的颜色,然后用QPainter在原来的像素上绘制新的颜色点,应该可行.

Qt之HTTP上传/下载

简述 在前面章节中我们讲述了关于Qt显示网络图片的内容,比较简单,因为图片一般都比较小,下载到本地速度比较快,所以基本不需要什么特殊处理,本节我们主要针对HTTP实现上传/下载进行详细的讲解与分享,包括:用户认证,实时获取下载大小.速度.剩余时间信息等. 首先看一下即将用到的公式: 文件剩余大小 = 文件总大小 - 文件已下载大小 平均速度 = 文件已下载大小 / 文件已下载大小所用的时间 瞬时速度 = 每秒下载的文件大小 剩余时间 = 文件剩余大小 / 瞬时速度 下面以下载为例,来实现一个文件

ile ystem odel-qt 显示本地文件 在treeView添加复选框

问题描述 qt 显示本地文件 在treeView添加复选框 QFileSystemModel *model = new QFileSystemModel(this);model->setFilter(QDir::Dirs | QDir::NoDotAndDotDot);model->setRootPath(""/""); ui->treeWidget->setModel(model);如何在文件夹前面添加复选框? 解决方案 model->