| Index: scons-2.0.1/engine/SCons/Tool/qt.py
|
| ===================================================================
|
| --- scons-2.0.1/engine/SCons/Tool/qt.py (revision 0)
|
| +++ scons-2.0.1/engine/SCons/Tool/qt.py (revision 0)
|
| @@ -0,0 +1,336 @@
|
| +
|
| +"""SCons.Tool.qt
|
| +
|
| +Tool-specific initialization for Qt.
|
| +
|
| +There normally shouldn't be any need to import this module directly.
|
| +It will usually be imported through the generic SCons.Tool.Tool()
|
| +selection method.
|
| +
|
| +"""
|
| +
|
| +#
|
| +# Copyright (c) 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2010 The SCons Foundation
|
| +#
|
| +# Permission is hereby granted, free of charge, to any person obtaining
|
| +# a copy of this software and associated documentation files (the
|
| +# "Software"), to deal in the Software without restriction, including
|
| +# without limitation the rights to use, copy, modify, merge, publish,
|
| +# distribute, sublicense, and/or sell copies of the Software, and to
|
| +# permit persons to whom the Software is furnished to do so, subject to
|
| +# the following conditions:
|
| +#
|
| +# The above copyright notice and this permission notice shall be included
|
| +# in all copies or substantial portions of the Software.
|
| +#
|
| +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY
|
| +# KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
| +# WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
| +# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
| +# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
| +# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
| +# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
| +#
|
| +
|
| +__revision__ = "src/engine/SCons/Tool/qt.py 5134 2010/08/16 23:02:40 bdeegan"
|
| +
|
| +import os.path
|
| +import re
|
| +
|
| +import SCons.Action
|
| +import SCons.Builder
|
| +import SCons.Defaults
|
| +import SCons.Scanner
|
| +import SCons.Tool
|
| +import SCons.Util
|
| +
|
| +class ToolQtWarning(SCons.Warnings.Warning):
|
| + pass
|
| +
|
| +class GeneratedMocFileNotIncluded(ToolQtWarning):
|
| + pass
|
| +
|
| +class QtdirNotFound(ToolQtWarning):
|
| + pass
|
| +
|
| +SCons.Warnings.enableWarningClass(ToolQtWarning)
|
| +
|
| +header_extensions = [".h", ".hxx", ".hpp", ".hh"]
|
| +if SCons.Util.case_sensitive_suffixes('.h', '.H'):
|
| + header_extensions.append('.H')
|
| +cplusplus = __import__('c++', globals(), locals(), [])
|
| +cxx_suffixes = cplusplus.CXXSuffixes
|
| +
|
| +def checkMocIncluded(target, source, env):
|
| + moc = target[0]
|
| + cpp = source[0]
|
| + # looks like cpp.includes is cleared before the build stage :-(
|
| + # not really sure about the path transformations (moc.cwd? cpp.cwd?) :-/
|
| + path = SCons.Defaults.CScan.path(env, moc.cwd)
|
| + includes = SCons.Defaults.CScan(cpp, env, path)
|
| + if not moc in includes:
|
| + SCons.Warnings.warn(
|
| + GeneratedMocFileNotIncluded,
|
| + "Generated moc file '%s' is not included by '%s'" %
|
| + (str(moc), str(cpp)))
|
| +
|
| +def find_file(filename, paths, node_factory):
|
| + for dir in paths:
|
| + node = node_factory(filename, dir)
|
| + if node.rexists():
|
| + return node
|
| + return None
|
| +
|
| +class _Automoc(object):
|
| + """
|
| + Callable class, which works as an emitter for Programs, SharedLibraries and
|
| + StaticLibraries.
|
| + """
|
| +
|
| + def __init__(self, objBuilderName):
|
| + self.objBuilderName = objBuilderName
|
| +
|
| + def __call__(self, target, source, env):
|
| + """
|
| + Smart autoscan function. Gets the list of objects for the Program
|
| + or Lib. Adds objects and builders for the special qt files.
|
| + """
|
| + try:
|
| + if int(env.subst('$QT_AUTOSCAN')) == 0:
|
| + return target, source
|
| + except ValueError:
|
| + pass
|
| + try:
|
| + debug = int(env.subst('$QT_DEBUG'))
|
| + except ValueError:
|
| + debug = 0
|
| +
|
| + # some shortcuts used in the scanner
|
| + splitext = SCons.Util.splitext
|
| + objBuilder = getattr(env, self.objBuilderName)
|
| +
|
| + # some regular expressions:
|
| + # Q_OBJECT detection
|
| + q_object_search = re.compile(r'[^A-Za-z0-9]Q_OBJECT[^A-Za-z0-9]')
|
| + # cxx and c comment 'eater'
|
| + #comment = re.compile(r'(//.*)|(/\*(([^*])|(\*[^/]))*\*/)')
|
| + # CW: something must be wrong with the regexp. See also bug #998222
|
| + # CURRENTLY THERE IS NO TEST CASE FOR THAT
|
| +
|
| + # The following is kind of hacky to get builders working properly (FIXME)
|
| + objBuilderEnv = objBuilder.env
|
| + objBuilder.env = env
|
| + mocBuilderEnv = env.Moc.env
|
| + env.Moc.env = env
|
| +
|
| + # make a deep copy for the result; MocH objects will be appended
|
| + out_sources = source[:]
|
| +
|
| + for obj in source:
|
| + if not obj.has_builder():
|
| + # binary obj file provided
|
| + if debug:
|
| + print "scons: qt: '%s' seems to be a binary. Discarded." % str(obj)
|
| + continue
|
| + cpp = obj.sources[0]
|
| + if not splitext(str(cpp))[1] in cxx_suffixes:
|
| + if debug:
|
| + print "scons: qt: '%s' is no cxx file. Discarded." % str(cpp)
|
| + # c or fortran source
|
| + continue
|
| + #cpp_contents = comment.sub('', cpp.get_text_contents())
|
| + cpp_contents = cpp.get_text_contents()
|
| + h=None
|
| + for h_ext in header_extensions:
|
| + # try to find the header file in the corresponding source
|
| + # directory
|
| + hname = splitext(cpp.name)[0] + h_ext
|
| + h = find_file(hname, (cpp.get_dir(),), env.File)
|
| + if h:
|
| + if debug:
|
| + print "scons: qt: Scanning '%s' (header of '%s')" % (str(h), str(cpp))
|
| + #h_contents = comment.sub('', h.get_text_contents())
|
| + h_contents = h.get_text_contents()
|
| + break
|
| + if not h and debug:
|
| + print "scons: qt: no header for '%s'." % (str(cpp))
|
| + if h and q_object_search.search(h_contents):
|
| + # h file with the Q_OBJECT macro found -> add moc_cpp
|
| + moc_cpp = env.Moc(h)
|
| + moc_o = objBuilder(moc_cpp)
|
| + out_sources.append(moc_o)
|
| + #moc_cpp.target_scanner = SCons.Defaults.CScan
|
| + if debug:
|
| + print "scons: qt: found Q_OBJECT macro in '%s', moc'ing to '%s'" % (str(h), str(moc_cpp))
|
| + if cpp and q_object_search.search(cpp_contents):
|
| + # cpp file with Q_OBJECT macro found -> add moc
|
| + # (to be included in cpp)
|
| + moc = env.Moc(cpp)
|
| + env.Ignore(moc, moc)
|
| + if debug:
|
| + print "scons: qt: found Q_OBJECT macro in '%s', moc'ing to '%s'" % (str(cpp), str(moc))
|
| + #moc.source_scanner = SCons.Defaults.CScan
|
| + # restore the original env attributes (FIXME)
|
| + objBuilder.env = objBuilderEnv
|
| + env.Moc.env = mocBuilderEnv
|
| +
|
| + return (target, out_sources)
|
| +
|
| +AutomocShared = _Automoc('SharedObject')
|
| +AutomocStatic = _Automoc('StaticObject')
|
| +
|
| +def _detect(env):
|
| + """Not really safe, but fast method to detect the QT library"""
|
| + QTDIR = None
|
| + if not QTDIR:
|
| + QTDIR = env.get('QTDIR',None)
|
| + if not QTDIR:
|
| + QTDIR = os.environ.get('QTDIR',None)
|
| + if not QTDIR:
|
| + moc = env.WhereIs('moc')
|
| + if moc:
|
| + QTDIR = os.path.dirname(os.path.dirname(moc))
|
| + SCons.Warnings.warn(
|
| + QtdirNotFound,
|
| + "Could not detect qt, using moc executable as a hint (QTDIR=%s)" % QTDIR)
|
| + else:
|
| + QTDIR = None
|
| + SCons.Warnings.warn(
|
| + QtdirNotFound,
|
| + "Could not detect qt, using empty QTDIR")
|
| + return QTDIR
|
| +
|
| +def uicEmitter(target, source, env):
|
| + adjustixes = SCons.Util.adjustixes
|
| + bs = SCons.Util.splitext(str(source[0].name))[0]
|
| + bs = os.path.join(str(target[0].get_dir()),bs)
|
| + # first target (header) is automatically added by builder
|
| + if len(target) < 2:
|
| + # second target is implementation
|
| + target.append(adjustixes(bs,
|
| + env.subst('$QT_UICIMPLPREFIX'),
|
| + env.subst('$QT_UICIMPLSUFFIX')))
|
| + if len(target) < 3:
|
| + # third target is moc file
|
| + target.append(adjustixes(bs,
|
| + env.subst('$QT_MOCHPREFIX'),
|
| + env.subst('$QT_MOCHSUFFIX')))
|
| + return target, source
|
| +
|
| +def uicScannerFunc(node, env, path):
|
| + lookout = []
|
| + lookout.extend(env['CPPPATH'])
|
| + lookout.append(str(node.rfile().dir))
|
| + includes = re.findall("<include.*?>(.*?)</include>", node.get_text_contents())
|
| + result = []
|
| + for incFile in includes:
|
| + dep = env.FindFile(incFile,lookout)
|
| + if dep:
|
| + result.append(dep)
|
| + return result
|
| +
|
| +uicScanner = SCons.Scanner.Base(uicScannerFunc,
|
| + name = "UicScanner",
|
| + node_class = SCons.Node.FS.File,
|
| + node_factory = SCons.Node.FS.File,
|
| + recursive = 0)
|
| +
|
| +def generate(env):
|
| + """Add Builders and construction variables for qt to an Environment."""
|
| + CLVar = SCons.Util.CLVar
|
| + Action = SCons.Action.Action
|
| + Builder = SCons.Builder.Builder
|
| +
|
| + env.SetDefault(QTDIR = _detect(env),
|
| + QT_BINPATH = os.path.join('$QTDIR', 'bin'),
|
| + QT_CPPPATH = os.path.join('$QTDIR', 'include'),
|
| + QT_LIBPATH = os.path.join('$QTDIR', 'lib'),
|
| + QT_MOC = os.path.join('$QT_BINPATH','moc'),
|
| + QT_UIC = os.path.join('$QT_BINPATH','uic'),
|
| + QT_LIB = 'qt', # may be set to qt-mt
|
| +
|
| + QT_AUTOSCAN = 1, # scan for moc'able sources
|
| +
|
| + # Some QT specific flags. I don't expect someone wants to
|
| + # manipulate those ...
|
| + QT_UICIMPLFLAGS = CLVar(''),
|
| + QT_UICDECLFLAGS = CLVar(''),
|
| + QT_MOCFROMHFLAGS = CLVar(''),
|
| + QT_MOCFROMCXXFLAGS = CLVar('-i'),
|
| +
|
| + # suffixes/prefixes for the headers / sources to generate
|
| + QT_UICDECLPREFIX = '',
|
| + QT_UICDECLSUFFIX = '.h',
|
| + QT_UICIMPLPREFIX = 'uic_',
|
| + QT_UICIMPLSUFFIX = '$CXXFILESUFFIX',
|
| + QT_MOCHPREFIX = 'moc_',
|
| + QT_MOCHSUFFIX = '$CXXFILESUFFIX',
|
| + QT_MOCCXXPREFIX = '',
|
| + QT_MOCCXXSUFFIX = '.moc',
|
| + QT_UISUFFIX = '.ui',
|
| +
|
| + # Commands for the qt support ...
|
| + # command to generate header, implementation and moc-file
|
| + # from a .ui file
|
| + QT_UICCOM = [
|
| + CLVar('$QT_UIC $QT_UICDECLFLAGS -o ${TARGETS[0]} $SOURCE'),
|
| + CLVar('$QT_UIC $QT_UICIMPLFLAGS -impl ${TARGETS[0].file} '
|
| + '-o ${TARGETS[1]} $SOURCE'),
|
| + CLVar('$QT_MOC $QT_MOCFROMHFLAGS -o ${TARGETS[2]} ${TARGETS[0]}')],
|
| + # command to generate meta object information for a class
|
| + # declarated in a header
|
| + QT_MOCFROMHCOM = (
|
| + '$QT_MOC $QT_MOCFROMHFLAGS -o ${TARGETS[0]} $SOURCE'),
|
| + # command to generate meta object information for a class
|
| + # declarated in a cpp file
|
| + QT_MOCFROMCXXCOM = [
|
| + CLVar('$QT_MOC $QT_MOCFROMCXXFLAGS -o ${TARGETS[0]} $SOURCE'),
|
| + Action(checkMocIncluded,None)])
|
| +
|
| + # ... and the corresponding builders
|
| + uicBld = Builder(action=SCons.Action.Action('$QT_UICCOM', '$QT_UICCOMSTR'),
|
| + emitter=uicEmitter,
|
| + src_suffix='$QT_UISUFFIX',
|
| + suffix='$QT_UICDECLSUFFIX',
|
| + prefix='$QT_UICDECLPREFIX',
|
| + source_scanner=uicScanner)
|
| + mocBld = Builder(action={}, prefix={}, suffix={})
|
| + for h in header_extensions:
|
| + act = SCons.Action.Action('$QT_MOCFROMHCOM', '$QT_MOCFROMHCOMSTR')
|
| + mocBld.add_action(h, act)
|
| + mocBld.prefix[h] = '$QT_MOCHPREFIX'
|
| + mocBld.suffix[h] = '$QT_MOCHSUFFIX'
|
| + for cxx in cxx_suffixes:
|
| + act = SCons.Action.Action('$QT_MOCFROMCXXCOM', '$QT_MOCFROMCXXCOMSTR')
|
| + mocBld.add_action(cxx, act)
|
| + mocBld.prefix[cxx] = '$QT_MOCCXXPREFIX'
|
| + mocBld.suffix[cxx] = '$QT_MOCCXXSUFFIX'
|
| +
|
| + # register the builders
|
| + env['BUILDERS']['Uic'] = uicBld
|
| + env['BUILDERS']['Moc'] = mocBld
|
| + static_obj, shared_obj = SCons.Tool.createObjBuilders(env)
|
| + static_obj.add_src_builder('Uic')
|
| + shared_obj.add_src_builder('Uic')
|
| +
|
| + # We use the emitters of Program / StaticLibrary / SharedLibrary
|
| + # to scan for moc'able files
|
| + # We can't refer to the builders directly, we have to fetch them
|
| + # as Environment attributes because that sets them up to be called
|
| + # correctly later by our emitter.
|
| + env.AppendUnique(PROGEMITTER =[AutomocStatic],
|
| + SHLIBEMITTER=[AutomocShared],
|
| + LIBEMITTER =[AutomocStatic],
|
| + # Of course, we need to link against the qt libraries
|
| + CPPPATH=["$QT_CPPPATH"],
|
| + LIBPATH=["$QT_LIBPATH"],
|
| + LIBS=['$QT_LIB'])
|
| +
|
| +def exists(env):
|
| + return _detect(env)
|
| +
|
| +# Local Variables:
|
| +# tab-width:4
|
| +# indent-tabs-mode:nil
|
| +# End:
|
| +# vim: set expandtab tabstop=4 shiftwidth=4:
|
|
|
| Property changes on: scons-2.0.1/engine/SCons/Tool/qt.py
|
| ___________________________________________________________________
|
| Added: svn:eol-style
|
| + LF
|
|
|
|
|