Python Pmw 和 cx_Freeze?
我无法从使用 Pmw(Python mega widgets)的 python 程序中创建可执行文件。我使用 cx_Freeze (通过 gui 后端“Gui2Exe”)。搜索 Pmw 站点我发现这是由 Pmw 库在运行时检查模块的方式引起的,并且当您使用 py2exe 或类似程序时它不起作用,因为这些库位于 zip 文件中。更多信息可以在这里找到:http://pmw.sourceforge.net/doc/dynamicloader.html< /a> 因此,他们在该页面的“冻结 Pmw”下提供了一个解决方案,通过提供一个脚本来生成一个可以轻松冻结的独立 Pmw 模块。但是,该脚本使用已弃用的代码,并且不适用于 Python 2.6 +。我尝试修改它但没有成功。
编辑:我想提一下,仅用“re”替换“regex”是行不通的。
#!/usr/bin/env python
# Helper script when freezing Pmw applications. It concatenates all
# Pmw megawidget files into a single file, 'Pmw.py', in the current
# directory. The script must be called with one argument, being the
# path to the 'lib' directory of the required version of Pmw.
# To freeze a Pmw application, you will also need to copy the
# following files to the application directory before freezing:
#
# PmwBlt.py PmwColor.py
import os
import regsub
import string
import sys
# The order of these files is significant. Files which reference
# other files must appear later. Files may be deleted if they are not
# used.
files = [
'Dialog', 'TimeFuncs', 'Balloon', 'ButtonBox', 'EntryField',
'Group', 'LabeledWidget', 'MainMenuBar', 'MenuBar', 'MessageBar',
'MessageDialog', 'NoteBook', 'OptionMenu', 'PanedWidget', 'PromptDialog',
'RadioSelect', 'ScrolledCanvas', 'ScrolledField', 'ScrolledFrame',
'ScrolledListBox', 'ScrolledText', 'HistoryText', 'SelectionDialog',
'TextDialog', 'TimeCounter', 'AboutDialog', 'ComboBox', 'ComboBoxDialog',
'Counter', 'CounterDialog',
]
# Set this to 0 if you do not use any of the Pmw.Color functions:
needColor = 1
# Set this to 0 if you do not use any of the Pmw.Blt functions:
needBlt = 1
def expandLinks(path):
if not os.path.isabs(path):
path = os.path.join(os.getcwd(), path)
while 1:
if not os.path.islink(path):
break
dir = os.path.dirname(path)
path = os.path.join(dir, os.readlink(path))
return path
def mungeFile(file):
# Read the file and modify it so that it can be bundled with the
# other Pmw files.
file = 'Pmw' + file + '.py'
text = open(os.path.join(srcdir, file)).read()
text = regsub.gsub('import Pmw\>', '', text)
text = regsub.gsub('INITOPT = Pmw.INITOPT', '', text)
text = regsub.gsub('\<Pmw\.', '', text)
text = '\n' + ('#' * 70) + '\n' + '### File: ' + file + '\n' + text
return text
# Work out which version is being bundled.
file = sys.argv[0]
file = os.path.normpath(file)
file = expandLinks(file)
dir = os.path.dirname(file)
dir = expandLinks(dir)
dir = os.path.dirname(dir)
dir = expandLinks(dir)
dir = os.path.basename(dir)
version = string.replace(dir[4:], '_', '.')
# Code to import the Color module.
colorCode = """
import PmwColor
Color = PmwColor
del PmwColor
"""
# Code to import the Blt module.
bltCode = """
import PmwBlt
Blt = PmwBlt
del PmwBlt
"""
# Code used when not linking with PmwBlt.py.
ignoreBltCode = """
_bltImported = 1
_bltbusyOK = 0
"""
# Code to define the functions normally supplied by the dynamic loader.
extraCode = """
### Loader functions:
_VERSION = '%s'
def setversion(version):
if version != _VERSION:
raise ValueError, 'Dynamic versioning not available'
def setalphaversions(*alpha_versions):
if alpha_versions != ():
raise ValueError, 'Dynamic versioning not available'
def version(alpha = 0):
if alpha:
return ()
else:
return _VERSION
def installedversions(alpha = 0):
if alpha:
return ()
else:
return (_VERSION,)
"""
if '-noblt' in sys.argv:
sys.argv.remove('-noblt')
needBlt = 0
if '-nocolor' in sys.argv:
sys.argv.remove('-nocolor')
needColor = 0
if len(sys.argv) != 2:
print 'usage: bundlepmw.py [-noblt] [-nocolor] /path/to/Pmw/Pmw_X_X_X/lib'
sys.exit()
srcdir = sys.argv[1]
if os.path.exists('Pmw.py'):
print 'Pmw.py already exists. Remove it and try again.'
sys.exit()
outfile = open('Pmw.py', 'w')
if needColor:
outfile.write(colorCode)
if needBlt:
outfile.write(bltCode)
outfile.write(extraCode % version)
# Specially handle PmwBase.py file:
text = mungeFile('Base')
text = regsub.gsub('import PmwLogicalFont', '', text)
text = regsub.gsub('PmwLogicalFont._font_initialise', '_font_initialise', text)
outfile.write(text)
if not needBlt:
outfile.write(ignoreBltCode)
files.append('LogicalFont')
for file in files:
text = mungeFile(file)
outfile.write(text)
print ''
print ' Pmw.py has been created.'
if needColor or needBlt:
print ' Before running freeze, also copy the following file(s):'
if needBlt:
print ' ' + os.path.join(srcdir, 'PmwBlt.py')
if needColor:
print ' ' + os.path.join(srcdir, 'PmwColor.py')
I am unable to make an executable from my python program which uses Pmw (Python mega widgets). I use cx_Freeze (through gui backend "Gui2Exe"). Searching Pmw site I've found that it's caused by how the Pmw library checks for modules when ran and it doesn't work when you use py2exe or similar programs because the libraries are in a zip file. More info can be found here: http://pmw.sourceforge.net/doc/dynamicloader.html
So they give a solution in that page, under "Freezing Pmw", by providing a script which generates a single standalone Pmw module which you can easily freeze. However, that script uses deprecated code and won't work with Python 2.6 +. I've tried to modify it with no luck.
EDIT: I'd like to mention that just replacing 'regex' with 're' won't work.
#!/usr/bin/env python
# Helper script when freezing Pmw applications. It concatenates all
# Pmw megawidget files into a single file, 'Pmw.py', in the current
# directory. The script must be called with one argument, being the
# path to the 'lib' directory of the required version of Pmw.
# To freeze a Pmw application, you will also need to copy the
# following files to the application directory before freezing:
#
# PmwBlt.py PmwColor.py
import os
import regsub
import string
import sys
# The order of these files is significant. Files which reference
# other files must appear later. Files may be deleted if they are not
# used.
files = [
'Dialog', 'TimeFuncs', 'Balloon', 'ButtonBox', 'EntryField',
'Group', 'LabeledWidget', 'MainMenuBar', 'MenuBar', 'MessageBar',
'MessageDialog', 'NoteBook', 'OptionMenu', 'PanedWidget', 'PromptDialog',
'RadioSelect', 'ScrolledCanvas', 'ScrolledField', 'ScrolledFrame',
'ScrolledListBox', 'ScrolledText', 'HistoryText', 'SelectionDialog',
'TextDialog', 'TimeCounter', 'AboutDialog', 'ComboBox', 'ComboBoxDialog',
'Counter', 'CounterDialog',
]
# Set this to 0 if you do not use any of the Pmw.Color functions:
needColor = 1
# Set this to 0 if you do not use any of the Pmw.Blt functions:
needBlt = 1
def expandLinks(path):
if not os.path.isabs(path):
path = os.path.join(os.getcwd(), path)
while 1:
if not os.path.islink(path):
break
dir = os.path.dirname(path)
path = os.path.join(dir, os.readlink(path))
return path
def mungeFile(file):
# Read the file and modify it so that it can be bundled with the
# other Pmw files.
file = 'Pmw' + file + '.py'
text = open(os.path.join(srcdir, file)).read()
text = regsub.gsub('import Pmw\>', '', text)
text = regsub.gsub('INITOPT = Pmw.INITOPT', '', text)
text = regsub.gsub('\<Pmw\.', '', text)
text = '\n' + ('#' * 70) + '\n' + '### File: ' + file + '\n' + text
return text
# Work out which version is being bundled.
file = sys.argv[0]
file = os.path.normpath(file)
file = expandLinks(file)
dir = os.path.dirname(file)
dir = expandLinks(dir)
dir = os.path.dirname(dir)
dir = expandLinks(dir)
dir = os.path.basename(dir)
version = string.replace(dir[4:], '_', '.')
# Code to import the Color module.
colorCode = """
import PmwColor
Color = PmwColor
del PmwColor
"""
# Code to import the Blt module.
bltCode = """
import PmwBlt
Blt = PmwBlt
del PmwBlt
"""
# Code used when not linking with PmwBlt.py.
ignoreBltCode = """
_bltImported = 1
_bltbusyOK = 0
"""
# Code to define the functions normally supplied by the dynamic loader.
extraCode = """
### Loader functions:
_VERSION = '%s'
def setversion(version):
if version != _VERSION:
raise ValueError, 'Dynamic versioning not available'
def setalphaversions(*alpha_versions):
if alpha_versions != ():
raise ValueError, 'Dynamic versioning not available'
def version(alpha = 0):
if alpha:
return ()
else:
return _VERSION
def installedversions(alpha = 0):
if alpha:
return ()
else:
return (_VERSION,)
"""
if '-noblt' in sys.argv:
sys.argv.remove('-noblt')
needBlt = 0
if '-nocolor' in sys.argv:
sys.argv.remove('-nocolor')
needColor = 0
if len(sys.argv) != 2:
print 'usage: bundlepmw.py [-noblt] [-nocolor] /path/to/Pmw/Pmw_X_X_X/lib'
sys.exit()
srcdir = sys.argv[1]
if os.path.exists('Pmw.py'):
print 'Pmw.py already exists. Remove it and try again.'
sys.exit()
outfile = open('Pmw.py', 'w')
if needColor:
outfile.write(colorCode)
if needBlt:
outfile.write(bltCode)
outfile.write(extraCode % version)
# Specially handle PmwBase.py file:
text = mungeFile('Base')
text = regsub.gsub('import PmwLogicalFont', '', text)
text = regsub.gsub('PmwLogicalFont._font_initialise', '_font_initialise', text)
outfile.write(text)
if not needBlt:
outfile.write(ignoreBltCode)
files.append('LogicalFont')
for file in files:
text = mungeFile(file)
outfile.write(text)
print ''
print ' Pmw.py has been created.'
if needColor or needBlt:
print ' Before running freeze, also copy the following file(s):'
if needBlt:
print ' ' + os.path.join(srcdir, 'PmwBlt.py')
if needColor:
print ' ' + os.path.join(srcdir, 'PmwColor.py')
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
不幸的是,原始文件存在一些制表符和空格问题(只需查看
expandlinks
中的while
缩进即可)。我修复了这些缩进问题,用
re.sub
更改了regsub.gsub
,并使用字符串类型方法消除了string
导入。之后脚本完美运行:
这里有更正后的脚本:
Unfortunately, the original file has some problems with tabs and spaces (just look at the
while
indentation inexpandlinks
).I fixed those indentation problems, changed
regsub.gsub
withre.sub
and eliminated thestring
import, using the string type methods.After that the script worked perfect:
Here you have the corrected script: