glade+pygtk学习 (3)

这几天用pygtk和glade写了一个不大的应用,在ubuntu下开发,然后移植到Maemo平台。

学习的过程中发现了SimpleGladeApp.py这个module,可以更方便地以面向对象的方式使用libgrade。

还记得serverinfo这个例子吗?里面是这样得到一个widget的实例的:

self.wTree = gtk.glade.XML(self.gladefile, self.windowname)
self.logwindowview = self.wTree.get_widget("textview1")

用了SimpleGladeApp.py,直接用self.logwindowview就行了。

下面是改造serverinfo使用SimpleGladeApp.py的步骤:

1. 从 http://www.gnomefiles.org/app.php/SimpleGladeApp.py 下载SimpleGladeApp.py

2. 解压缩后把tepache这个脚本放到你的PATH环境变量指定的路径下,比如/usr/local/bin,然后加上执行权限。

3. 我们已经有现成的serverinfo.glade文件了,现在要做的是用tepache这个脚本解析glade文件来生成一个Python模块,里面是一些对应window或dialog等顶层widget的class,并实现他们的callback方法。命令如下(-o选项是为了不覆盖原来的serverinfo.py):

$ tepache serverinfo.glade -o serverinfo_new.py
written file serverinfo_new.py
$ ls
serverinfo.glade serverinfo_new.py serverinfo.py
serverinfo.gladep serverinfo_new.py.orig SimpleGladeApp.py


可以看到tepache生成了三个文件:serverinfo_new.py, serverinfo_new.py.orig, SimpleGladeApp.py
.orig文件很重要,tepache用这个文件保护你在serverinfo_new.py添加的代码,所以不要删除这个文件。

可以先运行serverinfo_new.py看看效果。

4. serverinfo_new.py的部分代码:

......
# Warning: Do not modify any context comment such as #--
# They are required to keep user's code
......
from SimpleGladeApp import SimpleGladeApp
from SimpleGladeApp import bindtextdomain

app_name = "serverinfo"
app_version = "0.0.1"

glade_dir = ""
locale_dir = ""

bindtextdomain(app_name, locale_dir)

class Serverinfo(SimpleGladeApp):

def __init__(self, path="serverinfo.glade",
root="serverinfo",
domain=app_name, **kwargs):
path = os.path.join(glade_dir, path)
SimpleGladeApp.__init__(self, path, root, domain, **kwargs)

#-- Serverinfo.new {
def new(self):
print "A new %s has been created" % self.__class__.__name__
#-- Serverinfo.new }

#-- Serverinfo custom methods {
# Write your own methods here
#-- Serverinfo custom methods }

#-- Serverinfo.on_serverinfo_destroy {
def on_serverinfo_destroy(self, widget, *args):
print "on_serverinfo_destroy called with self.%s" % widget.get_name()
#-- Serverinfo.on_serverinfo_destroy }
......
#-- main {

def main():
serverinfo = Serverinfo()
demodialog = Demodialog()

serverinfo.run()

if __name__ == "__main__":
main()

#-- main }

5. 现在就可以添加一些代码来实现serverinfo的功能了。注意不要修改形如#-- Demodialog custom methods { 这样的注释,否则如果界面(glade文件)有了改动,tepache就没法做patch了。

把原来的insert_row改成ServerInfo类的一个方法,放在#-- Serverinfo custom methods 注释之间。
goUrl方法也放在这里。 注意访问widget实例的方法已经由

host = self.wTree.get_widget("entry1").get_text()

变成

host = self.entry1.get_text()

main方法里的实例化Demodialog的语句要挪到button2的clicked事件响应方法里,不然程序一运行对话框就出现了,你刚才一定注意到了。

destroy响应方法里加入self.gtk_main_quit()以便窗口销毁时能够退出程序。

6. 现在运行serverinfo_new.py,原来serverinfo.py的功能基本上都有了。但有一个问题,对话框弹出后点击“确定”或“取消”按钮都不能关闭对话框。我的解决办法是不在glade设计这个对话框,完成在程序里用代码实现。

7. 注意每次重新设计了界面,修改了glade文件,都要运行tepache来patch生成的python模块,已做的修改不会被覆盖。

新的代码下载地址: http://guoyong.org/?dl=pygtk-learning-serverinfo-simplegladeapp.tar.bz2

学习使用Carbide.j

最近想体验一下用Carbide.j开发J2ME程序。
先从开发环境的搭建开始。J2SDK自然是必不可少,1.4.2或更高的版本都可以。虽然Carbide.j可以独立运行,但也可以和一些流行的IDE集成,我选择了自己比较习惯的Eclipse。但在版本选择上走了弯路:一开始使用的是最新版的Eclipse 3.2,按照Forum Nokia提供的Screencast实现了一个hello world的例子,但没法在emulator里运行。一启动例子程序emulator就退出了。百般试验以后,确定不是程序的问题,而是Eclipse的版本不对,换了3.1.2就ok了。其实Carbide.j的安装说明里要求的Eclipse的版本就是3.0或3.1。

开发环境搭建好了,就可以学习那个screencast了解如何使用Carbide.j开发J2ME程序了。
screencast的下载地址:
http://forum.nokia.com/info/sw.nokia.com/id/93021595-9c99-4b0e-87b6-4a0b79d16541/Screencast_Carbide_j.html
要下载你得有一个Forum Nokia的帐号。

glade+pygtk学习 (2)

按照计划,今天参照这篇文章 Building an Application with PyGTK and Glade 学习。第一次使用menu和toolbar,进一步学习了tree/list view的使用,昨天还不太会用的dialog也学会了。

下载源代码: pygtk-learning-pywine.tgz

昨天遗留的几个问题也解决了:
1. 首先完成了那个dialog。虽然没什么功能,就算是复习一下dialog的实现方法吧。
2. 把ServerInfo程序的treeview内容改成显示http的header了。

#get http headers, then insert into tree model
data = page.read()
headers = page.info()
for name in headers.keys():
insert_row(self.treemodel, None, name, headers[name])

3. 把页面内容转换成unicode字符。用到了re模块去取页面meta标记里的字符集设置。

import re, string

r = re.compile(r""".*content=['|"]?text/html;\s?charset=([a-zA-Z0-9\-]+)['|"]?.*""", re.S | re.I | re.M)
m = r.match(data)
if m:
charset = string.lower(m.group(1))
if charset != 'utf-8':
data = data.decode(charset, 'ignore').encode('utf-8', 'ignore')
else:
data = data.encode('utf-8', 'ignore')
self.logwindow.set_text(data)

4. 用TexBuffer的set_text方法替代了insert_at_cursor。

新ServerInfo程序的源代码: pygtk-learning-serverinfo.tgz

glade+pygtk学习

今天上午参照 A Beginner's Guide to Using pyGTK and Glade 这篇文章学习,感觉确实很好用。文章开始部分对比几种GUI开发库写得挺好,虽然是2003年写的。还有后面提到如何结合socket编程,正好要用到。

照着Demo实做了一遍,发现一些用法已经deprecated了:

  • gtk.TRUE - 用True就行了
  • gtk.mainquit - 换成gtk.main_quit
  • gtk.mainloop() - 换成gtk.main()

期间犯了个小错误,忘了在glade里为主窗口添加destroy信号,结果没法关闭窗口后没法退出程序。

显示含有中文的页面内容时会在下面这行代码报错,像是编码问题,先不深究了。

self.logwindow.insert_at_cursor(data, len(data))

自己用为Entry控件加了activate信号,响应回车事件。也准备试试如何显示对话框,还没有完成。

可以在这里下载Demo的源代码: http://guoyong.org/?dl=pygtk-demo.tgz

接下来计划按照这篇文章 [ http://www.learningpython.com/2006/05/30/building-an-application-with-pygtk-and-glade/ ]学习,然后准备自己动手写了。