forked from bpbible/bpbible
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbpbible.py
248 lines (199 loc) · 7.21 KB
/
bpbible.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
#!/usr/bin/env python
import os
import sys
import glob
import shutil
import tempfile
if hasattr(sys, "frozen"):
# py2exe may get paths wrong
if os.path.dirname(sys.argv[0]):
os.chdir(os.path.dirname(sys.argv[0]))
# allow user to drop customization scripts in custom
# these will be auto-run
# for example, for distributing small patches
if os.path.isdir("custom"):
for file in os.listdir("custom"):
if os.path.isfile("custom/%s" % file) and file.endswith(".py"):
execfile("custom/%s" % file)
from util.debug import dprint, MESSAGE, WARNING, is_debugging
def enable_faulthandler():
try:
# Faulthandler helps us to find the stack trace when BPBible segmentation faults.
import faulthandler
faulthandler.enable()
except ImportError:
dprint(WARNING, "Faulthandler not installed")
except AttributeError, e:
# Faulthandler encounters an AttributeError when built with Py2exe,
# because py2exe is missing the fileno method on standard input and output.
# We catch this to prevent a crash.
dprint(WARNING, "AttributeError encountered enabling faulthandler.", e)
enable_faulthandler()
dprint(MESSAGE, "Importing wx")
import wx
from util import osutils
def find_xulrunner_path():
if osutils.is_mac():
path = os.getcwd() + "/../MacOS/"
if os.path.exists(path):
return path
else:
return os.getcwd() + "/dist/BPBible.app/Contents/MacOS/"
if osutils.is_gtk():
xulrunner_path = (osutils.find_file_in_path("xulrunner") or
osutils.find_file_in_path("xulrunner-stub"))
if xulrunner_path:
return os.path.dirname(os.path.realpath(xulrunner_path))
path = os.path.join(os.getcwd(), "xulrunner")
if not os.path.isdir(path):
# XXX: Perhaps we should make this error handling a little more friendly?
sys.stderr.write("Unable to find XULRunner.\n")
sys.exit(1)
return path
xulrunner_path = find_xulrunner_path()
dprint(MESSAGE, "XULRunner path is", xulrunner_path)
if osutils.is_msw():
os.environ['PATH'] = xulrunner_path + ';' + os.environ['PATH']
dprint(MESSAGE, "importing wx.wc")
import wx.wc
dprint(MESSAGE, "/importing wx")
# make sure contribs can be imported...
import contrib
import config, guiconfig
from util import confparser
from util.configmgr import config_manager
import util.i18n
import gui.i18n
class MyApp(wx.App):
def Initialize(self):
# for py2exe
import mainframe
was_restarted = self.restarting
self.starting = self.restarting = False
config_manager.load()
util.i18n.initialize()
gui.i18n.initialize()
frame = self.res.LoadFrame(None, "MainFrame" )
if(frame == None):
wx.MessageBox("Could not load MainFrame from auifrm.xrc", \
"Fatal Error", style = wx.ICON_ERROR)
return False
frame.SetIcons(guiconfig.icons)
self.SetTopWindow(frame)
frame.Show(osutils.is_gtk())
if was_restarted:
frame.Raise()
def OnInit(self):
self.InitXULRunner()
self.FindXULRunnerVersion()
self.ShowSplashScreen()
self.starting = True
self.restarting = False
self.reload_restarting = False
dprint(MESSAGE, "App Init")
guiconfig.load_icons()
from wx import xrc
self.res = xrc.XmlResource(config.xrc_path+"auifrm.xrc" )
return True
def ShowSplashScreen(self):
if not config.show_splashscreen():
self.splash = None
return
picture = 'splashscreen.png'
bitmap = wx.BitmapFromImage(
wx.Image(config.graphics_path + picture)
)
self.splash = wx.SplashScreen(
bitmap,
wx.SPLASH_CENTRE_ON_SCREEN|wx.SPLASH_NO_TIMEOUT,
0,
None,
style=wx.FRAME_NO_TASKBAR|wx.BORDER_NONE
)
self.splash.Show()
self.splash.Raise()
def InitXULRunner(self):
dprint(MESSAGE, "Initialising XULRunner engine")
self.InitProfile()
wx.wc.WebControl.InitEngine(xulrunner_path)
# NOTE: DO NOT move this import into the main import section.
# Doing so causes InitEngine() above to fail when loading xul.dll.
import gui.webconnect_protocol_handler
wx.wc.RegisterProtocol("bpbible", gui.webconnect_protocol_handler.MasterProtocolHandler())
dprint(MESSAGE, "XULRunner engine initialised")
def InitProfile(self):
# We need to use a random (uniqute) temp file for the profile directory
# to prevent a race condition consuming one CPU when two instances of
# the app use the same profile directory.
profile_path = tempfile.mkdtemp(prefix='bpbp.', suffix='.tmp')
dprint(MESSAGE, "Initialising profile directory: " + profile_path)
wx.wc.WebControl.SetProfilePath(profile_path)
# We can't delete the profile directory on process shutdown as
# Mozilla is still holding onto references for it.
# So instead, we delete old directories on startup.
#
# Note that this will work with multiple versions of BPBible running
# at once, as the other version will not allow the profile to be deleted.
old_profile_paths = glob.glob(os.path.join(os.path.dirname(profile_path), 'bpbp.*.tmp'))
for path in old_profile_paths:
if path != profile_path:
shutil.rmtree(path, ignore_errors=True)
def FindXULRunnerVersion(self):
"""Find the XULRunner version from the XULRunner platform.ini config file.
This should be provided by XULRunner (and then wxWebConnect) through
the nsIXULAppInfo API, but it seems this is only possible when xulrunner
has been run from the command line and has an application.ini and
XRE_Main() has been called.
Since XRE_Main() is just looking up the value in the INI file,
I figure it can't hurt too much to do the same here.
"""
xulrunner_ini_file = os.path.join(xulrunner_path, "platform.ini")
config_parser = confparser.config()
config_parser.read(xulrunner_ini_file)
if config_parser.has_option("Build", "Milestone"):
config.xulrunner_version = config_parser.get("Build", "Milestone")[0]
dprint(MESSAGE,"XULRunner version is %s." % config.xulrunner_version)
def main():
# convenience for when packaged (i.e. py2exe or py2app during dev on mac)
# - we can supply the name of a .py file and it will execfile it
if len(sys.argv) > 1:
if sys.argv[1].endswith(".py"):
f = sys.argv.pop(1)
return execfile(f, globals(), globals())
inspection_imported = False
try:
from wx.lib.mixins.inspect import InspectionMixin
inspection_imported = True
except ImportError, e:
try:
from wx.lib.mixins.inspection import InspectionMixin
inspection_imported = True
except ImportError, e:
pass
if inspection_imported and is_debugging():
class InspectableApp(MyApp, InspectionMixin):
def OnInit(self):
return_value = super(InspectableApp, self).OnInit()
if not return_value:
return False
self.Init()
return True
app = InspectableApp(0)
else:
if is_debugging():
dprint(WARNING, "Could not load inspection")
app = MyApp(0)
# Don't display warnings in distributed code unless it is in debugging mode.
import warnings
warnings.simplefilter("error" if is_debugging() else "ignore")
dprint(MESSAGE, "App created")
guiconfig.app = app
while app.starting or app.restarting:
if app.reload_restarting:
import reload_util
reload(reload_util)
reload_util.reload_all()
app.Initialize()
app.MainLoop()
if __name__ == '__main__':
main()