# IfcOpenShell - IFC toolkit and geometry engine # Copyright (C) 2021 Dion Moult # # This file is part of IfcOpenShell. # # IfcOpenShell is free software: you can redistribute it and/or modify # it under the terms of the GNU Lesser General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # IfcOpenShell is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU Lesser General Public License for more details. # # You should have received a copy of the GNU Lesser General Public License # along with IfcOpenShell. If not, see . from typing import Union import ifcopenshell import ifcopenshell.api.aggregate import ifcopenshell.api.geometry import ifcopenshell.api.owner import ifcopenshell.guid import ifcopenshell.util.element import ifcopenshell.util.placement def assign_container( file: ifcopenshell.file, products: list[ifcopenshell.entity_instance], relating_structure: ifcopenshell.entity_instance, ) -> Union[ifcopenshell.entity_instance, None]: """Assigns products to be contained hierarchically in a space All physical IFC model elements must be part of a hierarchical tree called the "spatial decomposition", where large things are made up of smaller things. This tree always begins at an "IfcProject" and is then broken down using "decomposition" relationships, of which aggregation is the first relationship you will use. See ifcopenshell.api.aggregate.assign_object for more details about aggregation. The IfcProject will be "decomposed" into spatial structure elements. These are virtual spaces like stes, buildings, storeys, and spaces (i.e. rooms). You can't physically touch these spaces, but you can touch the products contained within these spaces. To state that a product is contained in a space, you will use a "containment" relationship. Containment is a very common relationship used to create the hierarchical spatial decomposition tree. For example, you might say that "This wall is on the third building storey", or "this table is in the living room space". The distinguishing factor between aggregation and containment is that aggregation occurs between objects of the same type (e.g. a large space is made up of smaller spaces), whereas containment is between two different types: explicitly saying that a physical product exists within a virtual space. Containment is critical in construction management, to know which objects are in which spaces, as often you would divide your construction schedule into storey by storey, or zone by zone. Containment is also critical in facility management, as it indicates through which space equipment may be accessed for maintenance purposes. As a product may only have a single location in the "spatial decomposition" tree, assigning an aggregate relationship will remove any previous aggregation, containment, or nesting relationships it may have. :param products: A list of physical IfcElements existing in the space. :param relating_structure: The IfcSpatialStructureElement element, such as IfcBuilding, IfcBuildingStorey, or IfcSpace that the element exists in. :return: The IfcRelContainedInSpatialStructure relationship instance or `None` if `products` was empty list. Example: .. code:: python project = ifcopenshell.api.root.create_entity(model, ifc_class="IfcProject") site = ifcopenshell.api.root.create_entity(model, ifc_class="IfcSite") building = ifcopenshell.api.root.create_entity(model, ifc_class="IfcBuilding") storey = ifcopenshell.api.root.create_entity(model, ifc_class="IfcBuildingStorey") space = ifcopenshell.api.root.create_entity(model, ifc_class="IfcSpace") # The project contains a site (note that project aggregation is a special case in IFC) ifcopenshell.api.aggregate.assign_object(model, products=[site], relating_object=project) # The site has a building, the building has a storey, and the storey has a space ifcopenshell.api.aggregate.assign_object(model, products=[building], relating_object=site) ifcopenshell.api.aggregate.assign_object(model, products=[storey], relating_object=building) ifcopenshell.api.aggregate.assign_object(model, products=[space], relating_object=storey) # Create a wall and furniture wall = ifcopenshell.api.root.create_entity(model, ifc_class="IfcWall") furniture = ifcopenshell.api.root.create_entity(model, ifc_class="IfcFurniture") # The wall is in the storey, and the furniture is in the space ifcopenshell.api.spatial.assign_container(model, products=[wall], relating_structure=storey) ifcopenshell.api.spatial.assign_container(model, products=[furniture], relating_structure=space) """ if not products: return products_set = set(products) structure_rel = next(iter(relating_structure.ContainsElements), None) previous_containers_rels: set[ifcopenshell.entity_instance] = set() products_without_containers: list[ifcopenshell.entity_instance] = [] products_with_containers: list[ifcopenshell.entity_instance] = [] # check if there is anything to change for product in products_set: product_rel = next(iter(product.ContainedInStructure), None) if product_rel is None: products_without_containers.append(product) continue # either structure_rel is None or product is part of different rel if product_rel != structure_rel: previous_containers_rels.add(product_rel) products_with_containers.append(product) # products with already assigned containers will be skipped products_to_change = products_without_containers + products_with_containers # nothing to change if not products_to_change: return structure_rel # can be either only aggregated or only contained at the same time ifcopenshell.api.aggregate.unassign_object(file, products=products_without_containers) # unassign elements from previous containers for rel in previous_containers_rels: related_elements = set(rel.RelatedElements) - products_set if related_elements: rel.RelatedElements = list(related_elements) ifcopenshell.api.owner.update_owner_history(file, element=rel) else: history = rel.OwnerHistory file.remove(rel) if history: ifcopenshell.util.element.remove_deep2(file, history) # assign elements to a new container if structure_rel: structure_rel.RelatedElements = list(set(structure_rel.RelatedElements) | products_set) ifcopenshell.api.owner.update_owner_history(file, element=structure_rel) else: structure_rel = file.create_entity( "IfcRelContainedInSpatialStructure", **{ "GlobalId": ifcopenshell.guid.new(), "OwnerHistory": ifcopenshell.api.owner.create_owner_history(file), "RelatedElements": list(products_set), "RelatingStructure": relating_structure, } ) # localize placement relative to a new container for affected products for product in products_to_change: placement = getattr(product, "ObjectPlacement", None) if placement and placement.is_a("IfcLocalPlacement"): ifcopenshell.api.geometry.edit_object_placement( file, product=product, matrix=ifcopenshell.util.placement.get_local_placement(product.ObjectPlacement), is_si=False, ) return structure_rel