compile.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. # This script is based on the one found on: https://www.pygame.org/wiki/Pygame2exe?parent=CookBook
  2. # This means that I(Tankernn) have not created it and I do not claim to own it.
  3. #
  4. # This will create a dist directory containing the executable file, all the data
  5. # directories. All Libraries will be bundled in executable file.
  6. #
  7. # Run the build process by entering 'compile.py' or
  8. # 'python compile.py' in a console prompt.
  9. #
  10. # To build exe, python, pygame, and py2exe have to be installed. After
  11. # building exe none of this libraries are needed.
  12. # Please Note have a backup file in a different directory as if it crashes you
  13. # will loose it all!(I lost 6 months of work because I did not do this)
  14. try:
  15. from distutils.core import setup
  16. import py2exe, pygame
  17. from modulefinder import Module
  18. import glob, fnmatch
  19. import sys, os, shutil
  20. import operator
  21. except ImportError, message:
  22. raise SystemExit, "Unable to load module. %s" % message
  23. #hack which fixes the pygame mixer and pygame font
  24. origIsSystemDLL = py2exe.build_exe.isSystemDLL # save the orginal before we edit it
  25. def isSystemDLL(pathname):
  26. # checks if the freetype and ogg dll files are being included
  27. if os.path.basename(pathname).lower() in ("libfreetype-6.dll", "libogg-0.dll","sdl_ttf.dll"): # "sdl_ttf.dll" added by arit.
  28. return 0
  29. return origIsSystemDLL(pathname) # return the orginal function
  30. py2exe.build_exe.isSystemDLL = isSystemDLL # override the default function with this one
  31. class pygame2exe(py2exe.build_exe.py2exe): #This hack make sure that pygame default font is copied: no need to modify code for specifying default font
  32. def copy_extensions(self, extensions):
  33. #Get pygame default font
  34. pygamedir = os.path.split(pygame.base.__file__)[0]
  35. pygame_default_font = os.path.join(pygamedir, pygame.font.get_default_font())
  36. #Add font to list of extension to be copied
  37. extensions.append(Module("pygame.font", pygame_default_font))
  38. py2exe.build_exe.py2exe.copy_extensions(self, extensions)
  39. class BuildExe:
  40. def __init__(self):
  41. #Name of starting .py
  42. self.script = "PythonInvaders.py"
  43. #Name of program
  44. self.project_name = "PythonInvaders"
  45. #Project url
  46. self.project_url = "https://github.com/Tankernn/PythonInvaders"
  47. #Version of program
  48. self.project_version = "1.0"
  49. #License of the program
  50. self.license = "MIT License"
  51. #Auhor of program
  52. self.author_name = "Tankernn"
  53. self.author_email = "frans@tankernn.eu"
  54. self.copyright = "Copyright (c) 2015 Tankernn."
  55. #Description
  56. self.project_description = "A python version of Space Invaders"
  57. #Icon file (None will use pygame default icon)
  58. self.icon_file = None
  59. #Extra files/dirs copied to game
  60. self.extra_datas = ['img/']
  61. #Extra/excludes python modules
  62. self.extra_modules = []
  63. self.exclude_modules = []
  64. #DLL Excludes
  65. self.exclude_dll = ['']
  66. #python scripts (strings) to be included, seperated by a comma
  67. self.extra_scripts = []
  68. #Zip file name (None will bundle files in exe instead of zip file)
  69. self.zipfile_name = None
  70. #Dist directory
  71. self.dist_dir ='dist'
  72. ## Code from DistUtils tutorial at http://wiki.python.org/moin/Distutils/Tutorial
  73. ## Originally borrowed from wxPython's setup and config files
  74. def opj(self, *args):
  75. path = os.path.join(*args)
  76. return os.path.normpath(path)
  77. def find_data_files(self, srcdir, *wildcards, **kw):
  78. # get a list of all files under the srcdir matching wildcards,
  79. # returned in a format to be used for install_data
  80. def walk_helper(arg, dirname, files):
  81. if '.svn' in dirname:
  82. return
  83. names = []
  84. lst, wildcards = arg
  85. for wc in wildcards:
  86. wc_name = self.opj(dirname, wc)
  87. for f in files:
  88. filename = self.opj(dirname, f)
  89. if fnmatch.fnmatch(filename, wc_name) and not os.path.isdir(filename):
  90. names.append(filename)
  91. if names:
  92. lst.append( (dirname, names ) )
  93. file_list = []
  94. recursive = kw.get('recursive', True)
  95. if recursive:
  96. os.path.walk(srcdir, walk_helper, (file_list, wildcards))
  97. else:
  98. walk_helper((file_list, wildcards),
  99. srcdir,
  100. [os.path.basename(f) for f in glob.glob(self.opj(srcdir, '*'))])
  101. return file_list
  102. def run(self):
  103. if os.path.isdir(self.dist_dir): #Erase previous destination dir
  104. shutil.rmtree(self.dist_dir)
  105. #Use the default pygame icon, if none given
  106. if self.icon_file == None:
  107. path = os.path.split(pygame.__file__)[0]
  108. self.icon_file = os.path.join(path, 'pygame.ico')
  109. #List all data files to add
  110. extra_datas = []
  111. for data in self.extra_datas:
  112. if os.path.isdir(data):
  113. extra_datas.extend(self.find_data_files(data, '*'))
  114. else:
  115. extra_datas.append(('.', [data]))
  116. setup(
  117. cmdclass = {'py2exe': pygame2exe},
  118. version = self.project_version,
  119. description = self.project_description,
  120. name = self.project_name,
  121. url = self.project_url,
  122. author = self.author_name,
  123. author_email = self.author_email,
  124. license = self.license,
  125. # targets to build
  126. windows = [{
  127. 'script': self.script,
  128. 'icon_resources': [(0, self.icon_file)],
  129. 'copyright': self.copyright
  130. }],
  131. options = {'py2exe': {'optimize': 2, 'bundle_files': 1, 'compressed': True, \
  132. 'excludes': self.exclude_modules, 'packages': self.extra_modules, \
  133. 'dll_excludes': self.exclude_dll,
  134. 'includes': self.extra_scripts} },
  135. zipfile = self.zipfile_name,
  136. data_files = extra_datas,
  137. dist_dir = self.dist_dir
  138. )
  139. if os.path.isdir('build'): #Clean up build dir
  140. shutil.rmtree('build')
  141. if __name__ == '__main__':
  142. if operator.lt(len(sys.argv), 2):
  143. sys.argv.append('py2exe')
  144. BuildExe().run() #Run generation
  145. raw_input("Press any key to continue") #Pause to let user see that things ends