图形界面
简介
Xwindow是工作站图形系统的工业标准,它有多种不同的实现,多数Linux系统中使用Xorg。
当然,Xwindow有悠久的历史和传统,不过那不在我们讨论的范围。您要注意的有两点。
- Xwindow和Xbox中的X意义是不同的,X只是W之后的一个字母,差不多应该这样理解,Xwindow是Window的接班人(注意,Window不是Windows)
- 同样,也不要吧Xwindow说成是Xwindows,那是一种亵渎!一切伟大的创造,都应得到应有的理解和尊重。
架构及原理
Xwindow使用服务器-客户端架构。无论本地图形界面还是远程图形界面,都以同样的流程工作。这样便不需要分别进行设计和维护。
1
2
3
本地X客户端┐ ┌键盘
远程X客户端┼ X协议 - X服务器 - 驱动程序 ┼鼠标
远程X客户端┘ └显示器
Xserver
Xwindow系统服务器端,通过驱动程序(硬件规范)来管理硬件资源。
例如:当我们移动鼠标时,通过驱动程序,向Xserver发送信息:
“向右移动200点,向上移动100点”(向右上移动);“按下左键”……
Xserver作出如下响应:
- 上一次鼠标停止的坐标为600,500
- 向右200,向上100。现在鼠标位于坐标800,600
- 坐标800,600处,为窗口Firefox的“关闭”按钮
- 根据预设动作,将“点击Firefox窗口的关闭按钮”翻译为“关闭窗口Firefox”
- 向X客户端Firefox发送一个“退出”消息
- Xserver通过显示子系统(显卡、显示器),全程显示鼠标的位置和移动
事实上,向程序发送“退出”信号,通常窗口管理器完成。为了描述方便,这里暂不区分。
Xclient
Xwindow系统客户端,通过X协议,实现与Xserver的交互。例如:
- Xclient(假设Firefox)接收Xserver的消息:输入焦点在地址栏的范围内,“linuxtoy.org”,回车
- Firefox根据预设动作,将这些消息识别为“打开链接linuxtoy.org”
- Firefox向域名服务器请求链接“linuxtoy.org”。域名服务器将这个请求转换为“http://linuxtoy/org/”和IP地址211.148.131.7,发送回Firefox
- Firefox将”http://linuxtoy.org/”显示在地址栏(向Xserver发送请求,在地址栏位置显示这个位置)
- Firefox向地址211.148.131.7请示显示页面
- Firefox将服务器发送回的页面显示在主窗口中
Xprotocal
Xwindow系统协议,Xserver和Xclient之间进行通信的规则。
窗口管理器
Window Manager是一种特殊的Xclient
使用窗口管理器时,Xserver并不直接与其他Xclient通信,而是通过WM中转,当一些消息被定义为WM指令时,它们会被拦截。例如Alt+F4关闭窗口、拖动标题栏……
消息“打开链接linuxtoy.org”,具体内容如下:
输入焦点在地址栏的范围内,“linuxtoy.org”,回车
Xserver并不能直接判断焦点,而是这样:
- Xserver向WM发送位置和点击的信息,WM根据当前的“焦点策略”确定激活(最上层)的窗口为Firefox
- Xserver将Firefox显示在最上层,高亮显示它的标题栏
- 在窗口Firefox内点击地址栏,或者Ctrl+L,Xserver将位置信息发送给WM,WM再发送给Firefox
- Firefox判断当前焦点后,显示一个闪动的文字输入光标
- Firefox将输入光标通过WM发送给Xserver,Xserver在屏幕相应位置进行显示
那么,“窗口管理器”到底能做些什么呢?其实它所做的只有一件事——管理窗口。例如:
- 最上层的窗口会把其他窗口挡住
- 它通常是一个“已激活窗口”,根据不同的“焦点策略”,窗口管理器确定被激活的窗口。激活窗口标题栏高亮显示,接收大部分的 键盘消息和窗口内的鼠标点击信息
- 为了美观和容易分辨,大多数窗口都要有标题栏和边框。为了方便,标题栏上还要有一些按钮,比如:最小化,最大化,关闭(这些按钮是窗口管理器请求的小窗口)
- 一个窗口可以在另一个窗口旁边显示,而不一定完全被遮挡。为了实现这一点,就要控制窗口显示的位置
- 为了控制窗口的显示位置,需要将整个屏幕用坐标描述,最好的办法是绘制一个填充整个屏幕的窗口,也就是根窗口。
- 因为根窗口是最大的,所以它可以严严实实的遮挡任何窗口,为了避免这一点,根窗口永远在最底层。这很形象的说明了为什么它叫作“根窗口”
- 根窗口不一定只有一个,大多数的窗口管理器可以使用“工作区”,来切换显示多个根窗口
- 根窗口固定位置上通常放置一些其他Xclient的窗口,例如底部面板,顶部面板,侧面板,程序启动图标
- 面板上又可以放一些其他的Xclient窗口,如任务条,启动栏,菜单……任务条可以以图表显示正在运行的任务,还可以作其他的杂活,像自动挂载USB设备……
启动流程
我们知道init是linux的根进程,是所有进程的父进程。同样,xinit是所有Xwindow进程的根进程。
Startx
startx可以在命令行下启动图形界面。执行startx命令时,实际执行这一命令:
1
xinit /etc/X11/xinit/xinitrc -- /etc/X11/xinit/xserverrc
根据脚本/etc/X11/xinit/xserverrc 启动Xserver,同时根据脚本/etc/X11/xinit/xinitrc 启动指定Xclient进程,例如窗口管理器
/etc/X11/xinit/xserverrc 以预设的参数运行程序 /usr/bin/X11/X
/etc/X11/xinit/xinitrc 脚本则指向/etc/X11/Xserssion,依次启动/etc/X11/Xsession.d目录中的脚本
- 您可以在在用户配置文件~/.Xsession中定义使用WM,它的优先级高于全局配置文件(对于GDM会话不起作用)
- startx启动时,并不会再进行身份认证。因为它启动的是/etc/X11/Xsession.d/gnome-session,而不是GDM会话
GDM会话
许多Linux系统启动时自动进入图形界面,不需要运行startx命令
在某些启动级别中,包含了gdm的启动脚本,例如:/etc/rc2.d/S13gdm
- 指向/etc/gdm/gdm-cdd.conf文件,加载预设视觉主题,启动/usr/lib/gdm/gdmgreeter(登录屏幕)
- 用户身份认证完成后,启动/etc/X11/default-display-manager这个文件中设定的默认窗口管理器/usr/sbin/gdm
gdm在启动时,会要求用户名和密码,也就是我们看到的登录屏幕(gdmgreeter)
/usr/share/xsessions目录下为所有可用登录会话的脚本
配置文件
X服务器
X服务器的主要配置文件为/etc/X11/xorg.conf
1
2
3
4
5
6
7
8
9
10
11
12
13
Section "ServerLayout" #布局
Identifier "Xorg Configured" #布局标识
Screen 0 "Default Screen" 0 0 #屏幕标识
InputDevice "Generic Keyboard" #键盘标识
InputDevice "Configured Mouse" #鼠标标识
EndSection
Section "Module"
Load "ddc"
Load "dbe"
Load "dri"
Load "extmod"
Load "glx"
默认分辨率为默认色深下的第一个分辨率
配置文件内部结构
1
2
3
4
5
6
7
8
9
10
11
12
/
├/ "ServerLayout" 布局
|├ "InputDevice" keyboard 键盘
|├ "InputDevice" mouse 鼠标
||
|└/ "Screen" 显示子系统
│ ├ "Monitor" 显示器
│ ├ "Device" videocard 显卡
|
|
├ "Files" 字体
└ "Module" 模块
X客户端
在/etc/X11/Xsession文件中可以发现下列内容
1
2
3
4
5
6
7
8
9
OPTIONFILE=/etc/X11/Xsession.options①
②
SYSRESOURCES=/etc/X11/Xresources
USRRESOURCES=$HOME/.Xresources
③
SYSSESSIONDIR=/etc/X11/Xsession.d
USERXSESSION=$HOME/.xsession
ALTUSERXSESSION=$HOME/.Xsession
ERRFILE=$HOME/.xsession-errors
① 设定X进程的启动参数。例如运行用户进程allow-user-xsession
② X资源文件。许多程序保留了X接口,允许X服务器管理一些视觉选项,例如窗口内的字体,配色等
③ X进程。可以设置一些启动时自动运行的程序,也可以用来设定自己的窗口管理器(窗口管理器和桌面环境或者登录管理器是无关的)
字体
freetype 渲染引擎
作为Xorg服务器的一个模块,freetype的功能包括读取Truetype字体信息,如大小、分辨率、编码等,并以之为依据渲染字体-freetype2.x相对于freetype1.x增加了抗锯齿等功能-(/etc/X11/xorg.conf 的Module字段中,可以选择字体渲染模块)
freetype只负责渲染字体。而查找字体,则可以由X服务器、X客户端或者字体服务器来完成。找到字体后,使用freetype引擎进行渲染
X核心字体X服务器根据X客户端的请求(字符编码),查找字体并进行渲染,然后显示
Xft字体X客户端自行查找字体并进行渲染,X服务器只负责显示。由于Xft字体的渲染在客户端完成,所以它可以动态的加载,而不需要随同X服务器一同启动
字体服务器当客户端请求字体时,X服务器将请求转发到字体服务器,由字体服务器查找字体,并使用freetype引擎渲染,将结果传回X服务器,X服务器进行显示
X核心字体
/etc/X11/xorg.conf中可以配置X核心字体的搜索路径
1
2
3
4
5
6
7
Section "Files"
FontPath "/usr/X11R6/lib/X11/fonts/misc/"
FontPath "/usr/X11R6/lib/X11/fonts/Type1/"
FontPath "/usr/X11R6/lib/X11/fonts/Speedo/"
FontPath "/usr/X11R6/lib/X11/fonts/100dpi/"
FontPath "/usr/X11R6/lib/X11/fonts/75dpi/"
EndSection
- 当X客户端向X服务器请求显示文字的时候,X服务器会按上面列表的先后顺序查找字体。显示中文时,如果第一个路径中的字体不包含中文,则查找下面的路径,直到发现中文字体
- 请将您便好的字体放在靠前的位置
要使安装的字体能够作为X核心字体使用,将字体的安装路径添加到上面的列表中,使用mkfontscale、mkfontdir扫描文件夹中的字体,并生成索引,就可以了(建议使用ttmkfdir生成fonts.scale,将其复制为fonts.dir)
字体的选择及显示风格,可以修改GTK1的配置文件,或者在Xresources文件中对程序单独进行定义
事实上,在我们的日常应用中,X核心字体环境并不常见,使用GTK1图形库的程序、某些类型的终端……
XFT字体
Xft字体相关选项在/etc/fonts/fonts.conf文件中配置
1
2
3
4
<dir>/usr/share/X11/fonts</dir>
<dir>/usr/share/fonts</dir>
<dir>/usr/local/share/fonts</dir>
<dir>~/.fonts</dir>
可以使用fc-cache命令,递归扫描上面目录中的字体(包括子文件夹中的字体),建立字体缓存
多数支持GTK2或者Qt图形库的X客户端能够使用Xft字体渲染技术
安装字体,只要将字体拷贝到以上任意目录,fc-cache -fv刷新字体缓存即可(选项:-f强制刷新;-v显示过程)
使用命令fc-list列出所有可用字体。字体的选择及显示风格,可以修改GTK2或者Qt的配置文件,建议使用图形界面配置
参考资料: