diff options
| author | nyamatongwe <unknown> | 2012-05-17 12:46:29 +1000 | 
|---|---|---|
| committer | nyamatongwe <unknown> | 2012-05-17 12:46:29 +1000 | 
| commit | bf8b35542c4be4b8eebb71ca2ad518da2e12b850 (patch) | |
| tree | eab094df23f84c33c7ed03f3412a63a060d31eac /qt/ScintillaEditPy/sepbuild.py | |
| parent | 29c92749f25f24795ecd9fec2b6705129ebbd654 (diff) | |
| download | scintilla-mirror-bf8b35542c4be4b8eebb71ca2ad518da2e12b850.tar.gz | |
Qt platform layer added. Based on an implementation from Jason Haslam
at Scientific Toolworks, Inc. with additions performed for Wingware.
Diffstat (limited to 'qt/ScintillaEditPy/sepbuild.py')
| -rw-r--r-- | qt/ScintillaEditPy/sepbuild.py | 312 | 
1 files changed, 312 insertions, 0 deletions
| diff --git a/qt/ScintillaEditPy/sepbuild.py b/qt/ScintillaEditPy/sepbuild.py new file mode 100644 index 000000000..b694f15b0 --- /dev/null +++ b/qt/ScintillaEditPy/sepbuild.py @@ -0,0 +1,312 @@ +import distutils.sysconfig +import getopt +import glob +import os +import platform +import shutil +import subprocess +import stat +import sys + +sys.path.append(os.path.join("..", "ScintillaEdit")) +import WidgetGen + +# Decide up front which platform, treat anything other than Windows or OS X as Linux +PLAT_WINDOWS = platform.system() == "Windows" +PLAT_DARWIN = platform.system() == "Darwin" +PLAT_LINUX = not (PLAT_DARWIN or PLAT_WINDOWS) + +def IsFileNewer(name1, name2): +	""" Returns whether file with name1 is newer than file with name2.  Returns 1 +	if name2 doesn't exist. """ + +	if not os.path.exists(name1): +		return 0 + +	if not os.path.exists(name2): +		return 1 + +	mod_time1 = os.stat(name1)[stat.ST_MTIME] +	mod_time2 = os.stat(name2)[stat.ST_MTIME] +	return (mod_time1 > mod_time2) + +def textFromRun(args): +	(stdoutdata, stderrdata) = subprocess.Popen(args, shell=True, stdout=subprocess.PIPE).communicate() +	return stdoutdata + +def runProgram(args, exitOnFailure): +	print(" ".join(args)) +	retcode = subprocess.call(" ".join(args), shell=True, stderr=subprocess.STDOUT) +	if retcode: +		print("Failed in " + " ".join(args) + " return code = " + str(retcode)) +		if exitOnFailure: +			sys.exit() + +def usage(): +	print("sepbuild.py [-h|--help][-c|--clean][-u|--underscore-names]") +	print("") +	print("Generate PySide wappers and build them.") +	print("") +	print("options:") +	print("") +	print("-c --clean remove all object and generated files") +	print("-b --pyside-base  Location of the PySide+Qt4 sandbox to use") +	print("-h --help  display this text") +	print("-d --debug=yes|no  force debug build (or non-debug build)") +	print("-u --underscore-names  use method_names consistent with GTK+ standards") + +modifyFunctionElement = """		<modify-function signature="%s">%s +		</modify-function> +""" + +injectCode = """ +			<inject-code class="target" position="beginning">%s +			</inject-code>""" + +injectCheckN = """ +				if (!cppArg%d) { +					PyErr_SetString(PyExc_ValueError, "Null string argument"); +					return 0; +				}""" + +def methodSignature(name, v, options): +	argTypes = "" +	p1Type = WidgetGen.cppAlias(v["Param1Type"]) +	if p1Type: +		argTypes = argTypes + p1Type +	p2Type = WidgetGen.cppAlias(v["Param2Type"]) +	if p2Type and v["Param2Type"] != "stringresult": +		if p1Type: +			argTypes = argTypes + ", " +		argTypes = argTypes + p2Type +	methodName = WidgetGen.normalisedName(name, options, v["FeatureType"]) +	constDeclarator = " const" if v["FeatureType"] == "get" else "" +	return methodName + "(" + argTypes + ")" + constDeclarator + +def printTypeSystemFile(f,out, options): +	for name in f.order: +		v = f.features[name] +		if v["Category"] != "Deprecated": +			feat = v["FeatureType"] +			if feat in ["fun", "get", "set"]: +				checks = "" +				if v["Param1Type"] == "string": +					checks = checks + (injectCheckN % 0) +				if v["Param2Type"] == "string": +					if v["Param1Type"] == "":	# Only arg 2 -> treat as first +						checks = checks + (injectCheckN % 0) +					else: +						checks = checks + (injectCheckN % 1) +				if checks: +					inject = injectCode % checks +					out.write(modifyFunctionElement % (methodSignature(name, v, options), inject)) +				#if v["Param1Type"] == "string": +				#	out.write("<string-xml>" + name + "</string-xml>\n") + +def doubleBackSlashes(s): +	# Quote backslashes so qmake does not produce warnings +	return s.replace("\\", "\\\\") + +class SepBuilder: +	def __init__(self): +		# Discover configuration parameters +		self.ScintillaEditIncludes = [".", "../ScintillaEdit", "../ScintillaEditBase", "../../include"] +		if PLAT_WINDOWS: +			self.MakeCommand = "nmake" +			self.MakeTarget = "release" +		else: +			self.MakeCommand = "make" +			self.MakeTarget = "" + +		if PLAT_DARWIN: +			self.QMakeOptions = "-spec macx-g++" +		else: +			self.QMakeOptions = "" + +		# Default to debug build if running in a debug build interpreter +		self.DebugBuild = hasattr(sys, 'getobjects') + +		# Python +		self.PyVersion = "%d.%d" % sys.version_info[:2] +		self.PyVersionSuffix = distutils.sysconfig.get_config_var("VERSION") +		self.PyIncludes = distutils.sysconfig.get_python_inc() +		self.PyPrefix = distutils.sysconfig.get_config_var("prefix") +		self.PyLibDir = distutils.sysconfig.get_config_var( +			("LIBDEST" if sys.platform == 'win32' else "LIBDIR")) + +		# Scintilla +		with open("../../version.txt") as f: +			version = f.read() +			self.ScintillaVersion = version[0] + '.' + version[1] + '.' + version[2] + +		# Qt default location from qmake +		self._SetQtIncludeBase(textFromRun("qmake -query QT_INSTALL_HEADERS").rstrip()) + +		# PySide default location +		# No standard for installing PySide development headers and libs on Windows so +		# choose /usr to be like Linux +		self._setPySideBase('\\usr' if PLAT_WINDOWS else '/usr') + +		self.ProInclude = "sepbuild.pri" + +		self.qtStyleInterface = True + +	def _setPySideBase(self, base): +		 +		self.PySideBase = base +		if PLAT_LINUX: +			self.PySideTypeSystem = textFromRun("pkg-config --variable=typesystemdir pyside").rstrip() +			self.PySideIncludeBase = textFromRun("pkg-config --variable=includedir pyside").rstrip() +			self.ShibokenIncludeBase = textFromRun("pkg-config --variable=includedir shiboken").rstrip() +		else: +			self.PySideTypeSystem = os.path.join(self.PySideBase, "share", "PySide", "typesystems") +			self.ShibokenIncludeBase = os.path.join(self.PySideBase, "include", "shiboken") +			self.PySideIncludeBase = os.path.join(self.PySideBase, "include", "PySide") + +		self.PySideIncludes = [ +			self.ShibokenIncludeBase, +			self.PySideIncludeBase, +			os.path.join(self.PySideIncludeBase, "QtCore"), +			os.path.join(self.PySideIncludeBase, "QtGui")] + +		self.PySideLibDir = os.path.join(self.PySideBase, "lib") +		self.AllIncludes = os.pathsep.join(self.QtIncludes + self.ScintillaEditIncludes + self.PySideIncludes) + +		self.ShibokenGenerator = "shiboken" +		# Is this still needed? It doesn't work with latest shiboken sources +		#if PLAT_DARWIN: +		#	# On OS X, can not automatically find Shiboken dylib so provide a full path +		#	self.ShibokenGenerator = os.path.join(self.PySideLibDir, "generatorrunner", "shiboken") + +	def generateAPI(self, args): +		os.chdir(os.path.join("..", "ScintillaEdit")) +		if not self.qtStyleInterface: +			args.insert(0, '--underscore-names') +		WidgetGen.main(args) +		f = WidgetGen.readInterface(False) +		os.chdir(os.path.join("..", "ScintillaEditPy")) +		options = {"qtStyle": self.qtStyleInterface} +		WidgetGen.Generate("typesystem_ScintillaEdit.xml.template", "typesystem_ScintillaEdit.xml", printTypeSystemFile, f, options) + +	def runGenerator(self): +		generatorrunner = "shiboken" +		for name in ('shiboken', 'generatorrunner'): +			if PLAT_WINDOWS: +				name += '.exe' +			name = os.path.join(self.PySideBase, "bin", name) +			if os.path.exists(name): +				generatorrunner = name +				break +				 +		args = [ +			generatorrunner, +			"--generator-set=" + self.ShibokenGenerator, +			"global.h ", +			"--avoid-protected-hack", +			"--enable-pyside-extensions", +			"--include-paths=" + self.AllIncludes, +			"--typesystem-paths=" + self.PySideTypeSystem, +			"--output-directory=.", +			"typesystem_ScintillaEdit.xml"] +		print(" ".join(args)) +		retcode = subprocess.call(" ".join(args), shell=True, stderr=subprocess.STDOUT) +		if retcode: +			print("Failed in generatorrunner", retcode) +			sys.exit() + +	def writeVariables(self): +		# Write variables needed into file to be included from project so it does not have to discover much +		with open(self.ProInclude, "w") as f: +			f.write("SCINTILLA_VERSION=" + self.ScintillaVersion + "\n") +			f.write("PY_VERSION=" + self.PyVersion + "\n") +			f.write("PY_VERSION_SUFFIX=" + self.PyVersionSuffix + "\n") +			f.write("PY_PREFIX=" + doubleBackSlashes(self.PyPrefix) + "\n") +			f.write("PY_INCLUDES=" + doubleBackSlashes(self.PyIncludes) + "\n") +			f.write("PY_LIBDIR=" + doubleBackSlashes(self.PyLibDir) + "\n") +			f.write("PYSIDE_INCLUDES=" + doubleBackSlashes(self.PySideIncludeBase) + "\n") +			f.write("PYSIDE_LIB=" + doubleBackSlashes(self.PySideLibDir) + "\n") +			f.write("SHIBOKEN_INCLUDES=" + doubleBackSlashes(self.ShibokenIncludeBase) + "\n") +			if self.DebugBuild: +				f.write("CONFIG += debug\n") +			else: +				f.write("CONFIG += release\n") + +	def make(self): +		runProgram(["qmake", self.QMakeOptions], exitOnFailure=True) +		runProgram([self.MakeCommand, self.MakeTarget], exitOnFailure=True) + +	def cleanEverything(self): +		self.generateAPI(["--clean"]) +		runProgram([self.MakeCommand, "distclean"], exitOnFailure=False) +		try: +			os.remove(self.ProInclude) +		except OSError: +			pass +		for logFile in glob.glob("*.log"): +			try: +				os.remove(logFile) +			except OSError: +				pass +		shutil.rmtree("debug", ignore_errors=True) +		shutil.rmtree("release", ignore_errors=True) +		shutil.rmtree("ScintillaEditPy", ignore_errors=True) + +	def buildEverything(self): +		cleanGenerated = False +		opts, args = getopt.getopt(sys.argv[1:], "hcdub", +					["help", "clean", "debug=",  +					"underscore-names", "pyside-base="]) +		for opt, arg in opts: +			if opt in ("-h", "--help"): +				usage() +				sys.exit() +			elif opt in ("-c", "--clean"): +				cleanGenerated = True +			elif opt in ("-d", "--debug"): +				self.DebugBuild = (arg == '' or arg.lower() == 'yes') +				if self.DebugBuild and sys.platform == 'win32': +					self.MakeTarget = 'debug' +			elif opt in ("-b", '--pyside-base'): +				self._SetQtIncludeBase(os.path.join(os.path.normpath(arg), 'include')) +				self._setPySideBase(os.path.normpath(arg)) +			elif opt in ("-u", "--underscore-names"): +				self.qtStyleInterface = False + +		if cleanGenerated: +			self.cleanEverything() +		else: +			self.writeVariables() +			self.generateAPI([""]) +			self.runGenerator() +			self.make() +			self.copyScintillaConstants() +			 +	def copyScintillaConstants(self): +		 +		orig = 'ScintillaConstants.py' +		dest = '../../bin/' + orig +		if IsFileNewer(dest, orig): +			return +		 +		f = open(orig, 'r') +		contents = f.read() +		f.close() +		 +		f = open(dest, 'w') +		f.write(contents) +		f.close() + +	def _SetQtIncludeBase(self, base): + +		self.QtIncludeBase = base +		self.QtIncludes = [self.QtIncludeBase] + [os.path.join(self.QtIncludeBase, sub) for sub in ["QtCore", "QtGui"]] +		# Set path so correct qmake is found +		path = os.environ.get('PATH', '').split(os.pathsep) +		qt_bin_dir = os.path.join(os.path.dirname(base), 'bin') +		if qt_bin_dir not in path: +			path.insert(0, qt_bin_dir) +			os.environ['PATH'] = os.pathsep.join(path) + +if __name__ == "__main__": +	sepBuild = SepBuilder() +	sepBuild.buildEverything() | 
