`
izuoyan
  • 浏览: 9296350 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

xeyes的重新实现--x系统及其Xlib以及和windows的异同

阅读更多

每一个X程序都对应一个x client,x server则只负责输入输出事件和请求,键盘和鼠标以及触摸屏之类的动作会被x server捕获,作为事件通过x协议传给x client,然后x client得到这些事件之后会根据它们做一些逻辑计算,然后得到一个请求通过x协议发给x server,此时x server就会根据x client的请求将图形绘制出来,通过输出设备显示。一个x server可以同时被很多个x client连接,因此它必须管理总体的图形布局,包括哪个窗口属于哪个x client,哪个窗口此时位于最上方,哪个窗口被遮住了哪个部分等等,这些同样作为事件--类似键盘鼠标等的输入事件传给相应的x client,因此可以想象,每一个可以显示的元素都要帮定一个或者几个x client,x client得到这些事件(管理总体布局的事件,而非输入设备的事件)之后,同样要做出一些反应,比如当它的窗口重新显示的时候要重画整个它的窗口等等。
x系统是一个c/s结构的系统,它并不以窗口为核心,而是以进程(x client--xterm/xeyes,x server--xorg...)为核心的,x server进程负责捕获事件--包括输入设备的事件和窗口管理事件,并且将事件传递给相应的x client,而x client则根据事件负责逻辑处理,并将处理结果作为显示请求回传给x server,x server得到请求后进行显示。一般的,每一个x server都有一个事件队列和一个请求队列,每一个事件元素绑定x client,它的大体结构应该包含下面的逻辑:
1.
while(...) {
1.从事件队列取出一个事件;
2.得到事件队应的x client;
3.将事件发给x client;
}
2.
while(...) {
1.从请求队列取出一个请求;
3.根据请求进行显示;
}
3.
3.0:根据鼠标的位置或者(以及)当前窗口得到一个或者多个对之感兴趣的x client用于下面3.1到3.x的事件构造;
3.1:鼠标回调:构造一个事件加入事件队列;
3.2:键盘回调:构造一个事件加入事件队列;
...
x client则同样拥有一个事件队列和一个请求队列,但是逻辑和x server正好相反。
作为一个例子,首先启动一个x server(将配置中的tcpip打开,否则别的机器没法连接),然后用xhost增加一个可以连接的主机,然后在该主机上设置DISPLAY环境变量,最后运行一个最简单的xeyes,然后到x server上移动一下鼠标,在x client上strace一下这个xeyes,希望得到的是当晃动鼠标的时候,strace会打印出tcpip连接的数据,可是结果却很乱,不停的select,不停的超时,不停的打印...看了xeyes的源码才知道,原来它使用的是timer,也就是说不管有没有事件,timer都会定期的到期,然后去select网络连接,也就是和x server的连接。这样岂不是很浪费资源,于是就想用最底层的xlib重写一下著名的xeyes,xlib可以说是比较底层的x编程库了,再往下就是直接用socket用x协议写代码了。顺便说一句,在xeyes的man手册中,有这么一句话:Xeyes watches what you do and reports to the Boss.然而它并没有向任何人报告你的一举一动,看了xeyes的代码后,突然觉得这是可以的,因为XQueryPointer函数可以返回足够的信息,它足以让你的Boss将你fire了,具体的方式就是在XQueryPointer之后以XQueryPointer的第四个参数child_return为参数调用XFetchName(我们简单一点说,实际上还有更多信息可以获取的),然后就可以得到鼠标当前在哪个窗口上,这样Boss就知道你在干什么了,这个信息很容易通过syslog发送出去的,不过前提是Boss的xeyes窗口必须隐藏,也就是必须隐藏那双眼睛。在下面的代码中,我将不处理报告Boss的逻辑,因为上面已经简单说过了。
下面就是事件循环版本的xeyes的源代码了,它除了xlib没有使用其余的库,也没有使用轮询式的timer,而是使用了事件中断的触发机制,只有在有鼠标移动事件的时候xeyes才会行动,再者,所有和数学相关的代码完全拷贝timer版的xeyes代码,下面就是源代码:
#include <X11/Xlib.h>
#include <X11/Xos.h>
#include <stdio.h>
#include <X11/IntrinsicP.h>
#include <X11/StringDefs.h>
#include <X11/extensions/shape.h>
#include <stdio.h>
#include <math.h>
#include "transform.h"
int main(void)
{
Display *disp = XOpenDisplay(NULL); //得到默认的display
Window win_root = DefaultRootWindow(disp); //如果不使用这个root,将得不到xeyes窗口之外的事件。
int screen = DefaultScreen(display);
int black = BlackPixel(display, screen);
int white = WhitePixel(display, screen);
Window win_eyes = XCreateSimpleWindow(disp, win_root, 0, 0, 150, 100, 3, black, white);
XSelectInput(disp, win_root, PointerMotionMask );
XSelectInput(disp, win_eyes, ExposureMask|PointerMotionMask );
XMapWindow(disp, win_eyes);
XGCValues xgcv1;
xgcv1.function = GXorReverse;
XtGCMask valuemask1 = GCFunction|GCForeground | GCBackground;
GC gc_eyeballs = XCreateGC(disp, win_root, 0, NULL);
XGCValues values_return;
XGetGCValues (disp, gc_eyeballs, valuemask1, &values_return);
Pixel fg = values_return.foreground;
Pixel bg = values_return.background;
xgcv1.foreground = bg;
xgcv1.background = fg; //注意这里创建了两个反转的画笔,作用在于画眼珠子之前先擦除原来的眼珠子
GC gc_balls = XCreateGC(disp, win_root, valuemask1, &xgcv1);
XGCValues xgcv;
Pixmap pim = XCreatePixmap (disp, win_eyes, 150, 100, 1);
XtGCMask valuemask = GCForeground | GCBackground;
GC gc_eyes = XCreateGC (disp, pim, valuemask, &xgcv);
XSetForeground (disp, gc_eyes, 0);
XFillRectangle (disp, pim, gc_eyes, 0, 0,150, 100);
XSetForeground (disp, gc_eyes, 1);
Window rep_root, rep_child;
int rep_rootx, rep_rooty;
unsigned int rep_mask;
int dx, dy;
TPoint mouse;
TPoint newpupil[2]; //记录当次的眼睛位置
TPoint oldpupil[2]; //记录上次的眼睛位置
Transform t; //涉及到数学的计算,全部拷贝原始的xeyes代码
SetTransform (&t,0, 150,100, 0, W_MIN_X, W_MAX_X, W_MIN_Y, W_MAX_Y);
while(1) {
XEvent report = {0};
XNextEvent(disp, &report);
switch (report.type){
case Expose:
if (report.xexpose.window == win_root) {
//一般不会执行到这里,因为我们取的是根窗口,除非创建一个全屏幕的窗口使之遮住根窗口,然后移动它。
} else { //以下的T打头的函数全部拷贝自原始代码
TFillArc (disp, pim, gc_eyes, &t,
EYE_X(0) - EYE_HWIDTH - EYE_THICK,
EYE_Y(0) - EYE_HHEIGHT - EYE_THICK,
EYE_WIDTH + EYE_THICK * 2.0,
EYE_HEIGHT + EYE_THICK * 2.0,
90 * 64, 360 * 64); //画左眼
TFillArc (disp, pim, gc_eyes, &t,
EYE_X(1) - EYE_HWIDTH - EYE_THICK,
EYE_Y(1) - EYE_HHEIGHT - EYE_THICK,
EYE_WIDTH + EYE_THICK * 2.0,
EYE_HEIGHT + EYE_THICK * 2.0,
90 * 64, 360 * 64); //画右眼
//裁剪窗口
XShapeCombineMask (disp, win_eyes, ShapeBounding, 0, 0, pim, ShapeSet);
TFillArc (disp, win_eyes, gc_eyeballs, &t,
EYE_X(0) - EYE_HWIDTH - EYE_THICK,
EYE_Y(0) - EYE_HHEIGHT - EYE_THICK,
EYE_WIDTH + EYE_THICK * 2.0,
EYE_HEIGHT + EYE_THICK * 2.0,
90*64, 360 * 64); //涂黑左眼
TFillArc (disp, win_eyes, gc_balls, &t,
EYE_X(0) - EYE_HWIDTH,
EYE_Y(0) - EYE_HHEIGHT,
EYE_WIDTH, EYE_HEIGHT,
90 * 64, 360 * 64); //描左眼圈
TFillArc (disp, win_eyes, gc_eyeballs, &t,
EYE_X(1) - EYE_HWIDTH - EYE_THICK,
EYE_Y(1) - EYE_HHEIGHT - EYE_THICK,
EYE_WIDTH + EYE_THICK * 2.0,
EYE_HEIGHT + EYE_THICK * 2.0,
90*64, 360 * 64); //涂黑右眼
TFillArc (disp, win_eyes, gc_balls, &t,
EYE_X(1) - EYE_HWIDTH,
EYE_Y(1) - EYE_HHEIGHT,
EYE_WIDTH, EYE_HEIGHT,
90*64, 360 * 64); //描右眼圈
} //不要break,继续画眼珠子
case MotionNotify:
//得到当前鼠标的位置,还顺便得到了它所悬停的窗口信息
XQueryPointer (disp, win_root, &rep_root, &rep_child, &rep_rootx, &rep_rooty, &dx, &dy, &rep_mask);
mouse.x = Tx(dx, dy, &t);
mouse.y = Ty(dx, dy, &t);
computePupils (mouse, newpupil); //数学计算
TFillArc (disp, win_eyes, gc_balls, &t,
oldpupil[0].x - BALL_WIDTH / 2.0,
oldpupil[0].y - BALL_HEIGHT / 2.0,
BALL_WIDTH, BALL_HEIGHT,
90 * 64, 360 * 64); //先用反色清除掉原来的左眼珠子
TFillArc (disp, win_eyes, gc_eyeballs, &t,
newpupil[0].x - BALL_WIDTH / 2.0,
newpupil[0].y - BALL_HEIGHT / 2.0,
BALL_WIDTH, BALL_HEIGHT,
90 * 64, 360 * 64); //再用前景色填充左眼珠子
oldpupil[0].x = newpupil[0].x;
oldpupil[0].y = newpupil[0].y;
TFillArc (disp, win_eyes, gc_balls, &t,
oldpupil[1].x - BALL_WIDTH / 2.0,
oldpupil[1].y - BALL_HEIGHT / 2.0,
BALL_WIDTH, BALL_HEIGHT,
90 * 64, 360 * 64); //右眼珠子同左眼珠子
TFillArc (disp, win_eyes, gc_eyeballs, &t,
newpupil[1].x - BALL_WIDTH / 2.0,
newpupil[1].y - BALL_HEIGHT / 2.0,
BALL_WIDTH, BALL_HEIGHT,
90 * 64, 360 * 64);
oldpupil[1].x = newpupil[1].x;
oldpupil[1].y = newpupil[1].y;
break;
}
}
return 0;
}
Xlib编程很简单,只要几个要素齐备了,那么剩下的就是闭着眼睛敲代码了,首先是一个display--这是一个环境,然后是一个drawable--这是一张画布,它可以是一个windows,最后是一个GC--它是一支画笔,总的情况就是在display中用gc在drawable上作画,一个main函数框架就是:
main()
{
1.得到一个display,作为disp;
2.得到一个drawable,作为一个win;
3.得到一支画笔,作为gc;
4.注册自己感兴趣的事件告诉x server;
5.编写一个事件循环:
while (1) {
5.1.得到一个x server发来的事件;
5.2.处理该事件;
}
}
linux版本的无timer的xeyes到此结束,那么windows的呢?
可以想象,windows的实现更加简单一些,当用vc创建一个win32工程之后,IDE就已经帮你完成一半工作了,仔细理解这一半的工作,发现它和上述xlib编程的main框架是如此的相似。相似之后还有些许不同,这些不同完全是windows的消息机制和x系统的架构机制不同所导致的,x系统的当然要更简单一些,x系统基于进程,而windows基于线程,因此windows的消息机制之所以复杂,那是因为它使用一些操作系统本身的机制,比如线程。如果我们将windows的xeyes应用有意简化,使之向xlib的代码靠拢,最后我们发现,它们更加一致。

分享到:
评论

相关推荐

    xeyes-headtracking-js:Xeyes 2.0(javascript 和头部跟踪)

    在压缩包文件“xeyes-headtracking-js-master”中,包含了项目的源代码和其他相关资源。开发者可以通过查看和学习这些源码来理解头部跟踪的实现细节,包括如何整合上述步骤以及如何优化性能和用户体验。这不仅是一个...

    xorg-x11-apps-7.7-21.el8.x86-64.rpm xorg-x11-xbitmaps-1.1.1-6.el

    标题中的“xorg-x11-apps-7.7-21.el8.x86-64.rpm”和“xorg-x11-xbitmaps-1.1.1-6.el7.noarch.rpm”是两个在Linux环境中常见的软件包,它们与X Window系统(X11)相关,用于提供图形用户界面(GUI)应用程序和服务。...

    WpfEyes:XEyes 克隆最近的 Window 技术 (WPF)-开源

    WPF Eyes 是 Xeyes 的现代克隆,专为 Windows 操作系统设计,利用 WPF 提供的丰富图形和动画功能,为用户提供更加生动且交互性强的体验。 **WPF 技术详解** Windows Presentation Foundation(WPF)是 .NET ...

    basic_wm:基本X11窗口管理器示例

    建立和运行要构建此示例窗口管理器,您将需要: 启用C ++-11的C ++编译器 Xlib标头和库库要运行和测试它,您将需要: xinit 随机X实用程序(例如xclock , xeyes和xterm可与在 / ,您可以使用以下命令安装这些依赖项...

    [Ubuntu.Linux.基础视频教程].3.1.X.Window.System

    `xhost`命令控制X Server的访问权限,`xeyes`和`xlogo`等示例程序用于演示基本的X功能。 6. **安全和网络透明性**: X Window System支持网络透明性,意味着一个运行在远程主机上的应用程序可以显示在本地X Server...

    Xming-linux可视化工具,帮你安装更简便

    Xming是一款轻量级的开源X Window系统服务器,专为Windows和Linux平台设计,用于实现远程桌面功能。它允许你在没有图形界面的Linux系统上运行图形化应用,而无需安装完整的X Window系统。 Xming的工作原理是作为...

    Windows7配置PuTTY+Xming连接CentOS7 SSH服务器图形化界面

    在这个场景下,我们使用PuTTY作为SSH客户端,而Xming作为X Window System的本地实现,以实现远程Linux系统的图形界面在本地Windows上显示。下面将详细解释这两个工具以及如何配置它们。 PuTTY是一款免费、开源的SSH...

    Xming+fonts+putty

    本文将深入探讨这三个组件以及它们如何协同工作,为用户提供在Windows系统上访问远程Linux服务器的图形化界面。 首先,让我们逐一了解这三个工具: 1. **Xming**:Xming是一个开源的X Window系统服务器,它允许在...

    Linux配置Xmanager

    这个过程涉及了Linux系统的网络配置、X Window系统(X11)的工作原理以及如何安全地通过网络传输图形界面。 首先,要理解X Window系统是Unix和类Unix系统(如Linux)中的图形用户界面基础,它提供了窗口管理、图形...

    Linux的快捷键和主要命令.

    5. **&lt;Ctrl&gt;&lt;Alt&gt;(n=7到12)**:用于切换到第n个图形用户界面,通常情况下,从第7个终端开始运行X-windows,而第8至第12终端通常为空闲状态。 #### 三、常用命令 1. **帮助命令** - `man`: 查看命令的手册页。 - ...

    VcXsrv软件包(运行安装了WSL(Linux子系统Linux)的Linux GUI程序)

    VcXsrv是一个X服务器,能够接收来自远程X窗口系统(如Linux中的X Window System)的显示数据,并在本地Windows桌面环境下呈现出来。通过安装并配置VcXsrv,你可以轻松地在WSL上运行带有图形界面的Linux应用,如GIMP...

    xming+fonts+putty

    在IT领域,尤其是在远程连接和Linux系统操作中,"xming+fonts+putty"是一个常用的组合工具,用于在Windows操作系统上实现对Linux服务器的图形界面访问。这个组合中的每个组件都有其特定的功能和重要性。 首先,让...

    开源项目-disq-xgopher.zip

    今天我们将关注一个名为"disq-xgopher.zip"的开源项目,它内含一个名为"xgopher"的项目,是Xeyes的一个Go语言实现,赋予了经典的X Window System图形演示程序新的生命力。 Xeyes是一款早期的X Window System应用...

    Linux command summary

    ### Linux 常用命令总结 ...通过这些命令的学习与实践,我们可以更高效地管理和操作Linux系统中的文件与目录。以上是Linux常用命令的一些基本概述,希望能帮助到大家更好地理解和使用这些强大的工具。

    ubuntu个性化设置及有趣命令

    Ubuntu个性化设置及有趣命令的知识点解析: Ubuntu个性化设置: 1. 全局主题设置:通过修改系统...通过这些设置和命令,用户不仅能享受个性化定制的操作系统,还能在工作之余体验Linux系统中独特的趣味性和创造性。

    关于Linux跨网络运行XWindow程序

    总结起来,Linux跨网络运行XWindow程序涉及XWindow系统的基础原理、XProtocol通信协议、XServer和XClient的角色以及安全验证机制。熟练掌握这些知识点,不仅可以提高远程工作的效率,还能确保系统的安全性。正确配置...

    Linux命令大全(修改版)

    - X Window System提供了图形用户界面,常用命令如 `xeyes` 和 `xclock` 用于演示。 - `xdg-open` 打开文件或URL,根据其类型在默认的应用程序中打开。 12. **其他**: - `sudo` 以超级用户权限运行命令,提高...

    linux命令大全

    本资料集包含了广泛的命令,用于文件传输、备份解压、文件管理、磁盘管理、磁盘维护、系统设置、系统管理、文档编辑、网络通讯、电子邮件与新闻组、X Windows System以及其他的实用工具。 1. **文件传输**:`cp`和`...

Global site tag (gtag.js) - Google Analytics