diff --git a/pbxproj/pbxextensions/ProjectFiles.py b/pbxproj/pbxextensions/ProjectFiles.py index d470a90b..f55cd0cf 100644 --- a/pbxproj/pbxextensions/ProjectFiles.py +++ b/pbxproj/pbxextensions/ProjectFiles.py @@ -49,33 +49,33 @@ class ProjectFiles: _FILE_TYPES = { u'.a': (u'archive.ar', u'PBXFrameworksBuildPhase'), u'.app': (u'wrapper.application', None), - u'.s': (u'sourcecode.asm', u'PBXSourcesBuildPhase'), + u'.bundle': (u'wrapper.plug-in', u'PBXResourcesBuildPhase'), u'.c': (u'sourcecode.c.c', u'PBXSourcesBuildPhase'), u'.cpp': (u'sourcecode.cpp.cpp', u'PBXSourcesBuildPhase'), + u'.d': (u'sourcecode.dtrace', u'PBXSourcesBuildPhase'), + u'.dylib': (u'compiled.mach-o.dylib', u'PBXFrameworksBuildPhase'), u'.framework': (u'wrapper.framework', u'PBXFrameworksBuildPhase'), u'.h': (u'sourcecode.c.h', None), u'.hpp': (u'sourcecode.c.h', None), - u'.d': (u'sourcecode.dtrace', u'PBXSourcesBuildPhase'), - u'.swift': (u'sourcecode.swift', u'PBXSourcesBuildPhase'), u'.icns': (u'image.icns', u'PBXResourcesBuildPhase'), - u'.m': (u'sourcecode.c.objc', u'PBXSourcesBuildPhase'), u'.j': (u'sourcecode.c.objc', u'PBXSourcesBuildPhase'), + u'.json': (u'text.json', u'PBXResourcesBuildPhase'), + u'.m': (u'sourcecode.c.objc', u'PBXSourcesBuildPhase'), u'.mm': (u'sourcecode.cpp.objcpp', u'PBXSourcesBuildPhase'), u'.nib': (u'wrapper.nib', u'PBXResourcesBuildPhase'), u'.plist': (u'text.plist.xml', u'PBXResourcesBuildPhase'), - u'.json': (u'text.json', u'PBXResourcesBuildPhase'), u'.png': (u'image.png', u'PBXResourcesBuildPhase'), u'.rtf': (u'text.rtf', u'PBXResourcesBuildPhase'), + u'.s': (u'sourcecode.asm', u'PBXSourcesBuildPhase'), + u'.strings': (u'text.plist.strings', u'PBXResourcesBuildPhase'), + u'.swift': (u'sourcecode.swift', u'PBXSourcesBuildPhase'), + u'.tbd': (u'sourcecode.text-based-dylib-definition', u'PBXFrameworksBuildPhase'), u'.tiff': (u'image.tiff', u'PBXResourcesBuildPhase'), u'.txt': (u'text', u'PBXResourcesBuildPhase'), + u'.xcassets': (u'folder.assetcatalog', u'PBXResourcesBuildPhase'), + u'.xcdatamodeld': (u'wrapper.xcdatamodel', u'PBXSourcesBuildPhase'), u'.xcodeproj': (u'wrapper.pb-project', None), u'.xib': (u'file.xib', u'PBXResourcesBuildPhase'), - u'.strings': (u'text.plist.strings', u'PBXResourcesBuildPhase'), - u'.bundle': (u'wrapper.plug-in', u'PBXResourcesBuildPhase'), - u'.dylib': (u'compiled.mach-o.dylib', u'PBXFrameworksBuildPhase'), - u'.xcdatamodeld': (u'wrapper.xcdatamodel', u'PBXSourcesBuildPhase'), - u'.xcassets': (u'folder.assetcatalog', u'PBXResourcesBuildPhase'), - u'.tbd': (u'sourcecode.text-based-dylib-definition', u'PBXFrameworksBuildPhase'), } _SPECIAL_FOLDERS = [ u'.bundle', @@ -105,11 +105,8 @@ def add_file(self, path, parent=None, tree=TreeType.SOURCE_ROOT, target_name=Non """ results = [] # if it's not forced to add the file stop if the file already exists. - if not force: - for section in self.objects.get_sections(): - for obj in self.objects.get_objects_in_section(section): - if u'path' in obj and ProjectFiles._path_leaf(path) == ProjectFiles._path_leaf(obj.path): - return results + if not force and self._file_in_project(path): + return results # decide the proper tree and path to add abs_path, path, tree = ProjectFiles._get_path_and_tree(self._source_root, path, tree) @@ -133,6 +130,9 @@ def add_file(self, path, parent=None, tree=TreeType.SOURCE_ROOT, target_name=Non if not file_options.create_build_files: return results + # check if the file_ref should be processed as a variant group + file_ref = self._create_variant_groups(file_ref) + # additional attributes in for libraries/embed frameworks attributes = file_options.get_attributes() @@ -144,20 +144,9 @@ def add_file(self, path, parent=None, tree=TreeType.SOURCE_ROOT, target_name=Non # if it's a framework and it needs to be embedded if file_options.embed_framework and expected_build_phase == u'PBXFrameworksBuildPhase' and \ file_ref.lastKnownFileType == u'wrapper.framework': - embed_phase = target.get_or_create_build_phase(u'PBXCopyFilesBuildPhase', - (PBXCopyFilesBuildPhaseNames.EMBEDDED_FRAMEWORKS,)) - # add runpath search flag - self.add_flags(XCBuildConfigurationFlags.LD_RUNPATH_SEARCH_PATHS, - u'$(inherited) @executable_path/Frameworks', target_name) - build_phases.extend(embed_phase) - - # create the build file and add it to the phase - for target_build_phase in build_phases: - build_file = PBXBuildFile.create(file_ref, attributes) - self.objects[build_file.get_id()] = build_file - target_build_phase.add_build_file(build_file) + self._embed_framework(file_ref, target, build_phases) - results.append(build_file) + results.extend(self._create_build_files(file_ref, attributes, build_phases)) # special case for the frameworks and libraries to update the search paths if tree != TreeType.SOURCE_ROOT or abs_path is None: @@ -331,6 +320,57 @@ def add_folder(self, path, parent=None, excludes=None, recursive=True, create_gr return results + # private methods + def _create_build_files(self, file_ref, attributes, build_phases): + results = [] + # create the build file and add it to the phase + for target_build_phase in build_phases: + build_file = PBXBuildFile.create(file_ref, attributes) + self.objects[build_file.get_id()] = build_file + target_build_phase.add_build_file(build_file) + + results.append(build_file) + + return results + + def _embed_framework(self, file_ref, target, build_phases): + embed_phase = target.get_or_create_build_phase(u'PBXCopyFilesBuildPhase', + (PBXCopyFilesBuildPhaseNames.EMBEDDED_FRAMEWORKS,)) + # add runpath search flag + self.add_flags(XCBuildConfigurationFlags.LD_RUNPATH_SEARCH_PATHS, + u'$(inherited) @executable_path/Frameworks', target.name) + build_phases.extend(embed_phase) + + def _create_variant_groups(self, file_ref): + if file_ref.lastKnownFileType != u'text.plist.strings': + return file_ref + + variant_group = None + expected_name = os.path.split(file_ref.path)[1] + for variant in self.objects.get_objects_in_section(u'PBXVariantGroup'): + if variant.name == expected_name: + variant_group = variant + break + + if variant_group is None: + variant_group = PBXVariantGroup.create(expected_name) + self.objects[variant_group.get_id()] = variant_group + + variant_group.add_variant(file_ref) + return variant_group + + def _file_in_project(self, path): + """ + Checks if the path of the file already exists on the project on any section + :param path: Path of the file to check + :return: if the file exists on any section or not. + """ + for section in self.objects.get_sections(): + for obj in self.objects.get_objects_in_section(section): + if u'path' in obj and ProjectFiles._path_leaf(path) == ProjectFiles._path_leaf(obj.path): + return True + return False + # miscellaneous functions, candidates to be extracted and decouple implementation @classmethod def _determine_file_type(cls, file_ref, unknown_type_allowed): @@ -343,8 +383,8 @@ def _determine_file_type(cls, file_ref, unknown_type_allowed): if not unknown_type_allowed and file_type is None: raise ValueError( - u'Unknown file extension: {0}. Please add the extension and Xcode type to ProjectFiles._FILE_TYPES' \ - .format(os.path.splitext(file_ref.name)[1])) + u'Unknown file extension: {0}. Please add the extension and Xcode type to ProjectFiles._FILE_TYPES' + .format(os.path.splitext(file_ref.name)[1])) return file_type, build_phase @@ -368,4 +408,4 @@ def _get_path_and_tree(cls, source_root, path, tree): else: tree = TreeType.ABSOLUTE - return abs_path, path, tree + return abs_path, path, tree \ No newline at end of file diff --git a/pbxproj/pbxsections/PBXFileReference.py b/pbxproj/pbxsections/PBXFileReference.py index 2af4fc27..30956b60 100644 --- a/pbxproj/pbxsections/PBXFileReference.py +++ b/pbxproj/pbxsections/PBXFileReference.py @@ -23,6 +23,10 @@ def set_last_known_file_type(self, file_type): del self[u'explicitFileType'] self[u'lastKnownFileType'] = file_type + # for localization strings the name has to be the name of the language file + if file_type == u'text.plist.strings': + self.name = os.path.splitext(os.path.split(self.path)[0])[0] + def _print_object(self, indentation_depth=u'', entry_separator=u'\n', object_start=u'\n', indentation_increment=u'\t'): return super(PBXFileReference, self)._print_object(u'', entry_separator=u' ', object_start=u'', diff --git a/pbxproj/pbxsections/PBXGroup.py b/pbxproj/pbxsections/PBXGroup.py index 25cfcbed..b3f526e4 100644 --- a/pbxproj/pbxsections/PBXGroup.py +++ b/pbxproj/pbxsections/PBXGroup.py @@ -56,17 +56,3 @@ def remove_child(self, child): return True return False - - def remove(self, recursive=True): - parent = self.get_parent() - # remove from the objects reference - del parent[self.get_id()] - - # remove children if necessary - if recursive: - for subgroup_id in self.children: - subgroup = parent[subgroup_id] - if subgroup is None or not subgroup.remove(recursive): - return False - - return True diff --git a/pbxproj/pbxsections/PBXVariantGroup.py b/pbxproj/pbxsections/PBXVariantGroup.py new file mode 100644 index 00000000..3f5fdbdb --- /dev/null +++ b/pbxproj/pbxsections/PBXVariantGroup.py @@ -0,0 +1,30 @@ +from pbxproj.PBXGenericObject import * +from pbxproj.pbxsections.PBXFileReference import * + + +class PBXVariantGroup(PBXGenericObject): + @classmethod + def create(cls, name, children=None, source_tree=u''): + return cls().parse({ + u'_id': cls._generate_id(), + u'isa': cls.__name__, + u'children': children if children else [], + u'name': name, + u'sourceTree': source_tree + }) + + def add_variant(self, file_ref): + if not isinstance(file_ref, PBXFileReference): + return False + + self.children.append(file_ref.get_id()) + return True + + def remove_variant(self, file_ref): + if not isinstance(file_ref, PBXFileReference): + return False + + self.children.remove(file_ref.get_id()) + del self.get_parent()[file_ref.get_id()] + + return True diff --git a/pbxproj/pbxsections/__init__.py b/pbxproj/pbxsections/__init__.py index 816c0d9a..3a009724 100644 --- a/pbxproj/pbxsections/__init__.py +++ b/pbxproj/pbxsections/__init__.py @@ -14,3 +14,4 @@ from pbxproj.pbxsections.PBXAggregatedTarget import * from pbxproj.pbxsections.PBXNativeTarget import * from pbxproj.pbxsections.PBXGroup import * +from pbxproj.pbxsections.PBXVariantGroup import * diff --git a/tests/pbxextensions/TestProjectFiles.py b/tests/pbxextensions/TestProjectFiles.py index 57fc8ec5..9ea98107 100644 --- a/tests/pbxextensions/TestProjectFiles.py +++ b/tests/pbxextensions/TestProjectFiles.py @@ -143,6 +143,25 @@ def testAddFileIfExists(self): build_file = project.add_file("file.m", force=False) self.assertEqual(build_file, []) + def testAddVariantGroupFile(self): + project = XcodeProject(self.obj) + build_file = project.add_file("en.proj/Texts.strings") + + self.assertEqual(project.objects.get_objects_in_section(u'PBXVariantGroup').__len__(), 1) + self.assertEqual(build_file.__len__(), 2) + self.assertEqual(build_file[0].fileRef, project.objects.get_objects_in_section(u'PBXVariantGroup')[0].get_id()) + self.assertEqual(build_file[1].fileRef, project.objects.get_objects_in_section(u'PBXVariantGroup')[0].get_id()) + + def testAddVariantGroupFileToExistingGroup(self): + project = XcodeProject(self.obj) + build_file = project.add_file("en.proj/Texts.strings") + build_file = project.add_file("nl.proj/Texts.strings") + + self.assertEqual(project.objects.get_objects_in_section(u'PBXVariantGroup').__len__(), 1) + self.assertEqual(build_file.__len__(), 2) + self.assertEqual(build_file[0].fileRef, project.objects.get_objects_in_section(u'PBXVariantGroup')[0].get_id()) + self.assertEqual(build_file[1].fileRef, project.objects.get_objects_in_section(u'PBXVariantGroup')[0].get_id()) + def testGetFilesByNameWithNoParent(self): project = XcodeProject(self.obj) files = project.get_files_by_name('file') diff --git a/tests/pbxsections/TestPBXVariantGroup.py b/tests/pbxsections/TestPBXVariantGroup.py new file mode 100644 index 00000000..d98d848d --- /dev/null +++ b/tests/pbxsections/TestPBXVariantGroup.py @@ -0,0 +1,30 @@ +import unittest +from pbxproj.pbxsections.PBXVariantGroup import * +from pbxproj.PBXObjects import * + + +class PBXVariantGroupTest(unittest.TestCase): + def testAddVariantInvalidFileRef(self): + variant = PBXVariantGroup.create(u'texts') + self.assertFalse(variant.add_variant(PBXGenericObject().parse({u'isa': ''}))) + + def testAddVariantValidFileRef(self): + variant = PBXVariantGroup.create(u'texts') + self.assertTrue(variant.add_variant(PBXFileReference().parse({u'_id': '1', u'isa': 'PBXFileReference'}))) + self.assertIn('1', variant.children) + + def testRemoveVariantInvalidFileRef(self): + variant = PBXVariantGroup.create(u'texts') + self.assertFalse(variant.remove_variant(PBXGenericObject().parse({u'isa': ''}))) + + def testRemoveVariantValidFileRef(self): + objs = objects().parse({u'1': {u'isa': 'PBXFileReference'}}) + file_ref = objs['1'] + + variant = PBXVariantGroup.create(u'texts') + objs[variant.get_id()] = variant + + variant.add_variant(file_ref) + + self.assertTrue(variant.remove_variant(file_ref)) + self.assertNotIn('1', variant.children)