Claude-skill-registry clash-detection-analysis
Detect and analyze geometric clashes between BIM elements. Identify hard clashes, soft clashes, and workflow conflicts using spatial analysis and rule-based detection.
install
source · Clone the upstream repo
git clone https://github.com/majiayu000/claude-skill-registry
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/majiayu000/claude-skill-registry "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/data/clash-detection-analysis" ~/.claude/skills/majiayu000-claude-skill-registry-clash-detection-analysis && rm -rf "$T"
manifest:
skills/data/clash-detection-analysis/SKILL.mdsource content
Clash Detection Analysis
Overview
This skill implements automated clash detection for BIM models. Identify conflicts between building elements before construction to prevent costly rework and delays.
Types of Clashes:
- Hard Clash: Physical intersection of elements
- Soft Clash: Clearance/tolerance violations
- Workflow Clash: Scheduling/sequencing conflicts
"Обнаружение коллизий на этапе проектирования может сократить затраты на исправление ошибок до 10 раз по сравнению с исправлением на стройплощадке."
Quick Start
import ifcopenshell import ifcopenshell.geom import numpy as np from itertools import combinations # Open model ifc = ifcopenshell.open("model.ifc") # Get structural and MEP elements structural = ifc.by_type("IfcColumn") + ifc.by_type("IfcBeam") mep = ifc.by_type("IfcPipeSegment") + ifc.by_type("IfcDuctSegment") # Simple bounding box clash check settings = ifcopenshell.geom.settings() def get_bbox(element): try: shape = ifcopenshell.geom.create_shape(settings, element) verts = np.array(shape.geometry.verts).reshape(-1, 3) return verts.min(axis=0), verts.max(axis=0) except: return None, None def check_bbox_clash(bbox1, bbox2): min1, max1 = bbox1 min2, max2 = bbox2 if min1 is None or min2 is None: return False return np.all(max1 >= min2) and np.all(max2 >= min1) # Find clashes clashes = [] for s_elem in structural: for m_elem in mep: bbox1 = get_bbox(s_elem) bbox2 = get_bbox(m_elem) if check_bbox_clash(bbox1, bbox2): clashes.append({ 'element1': s_elem.GlobalId, 'element2': m_elem.GlobalId, 'type': 'Structure-MEP' }) print(f"Found {len(clashes)} potential clashes")
Clash Detection Engine
Core Detector Class
import ifcopenshell import ifcopenshell.geom import numpy as np import pandas as pd from dataclasses import dataclass from typing import List, Dict, Optional, Tuple from itertools import combinations from scipy.spatial import cKDTree @dataclass class Clash: element1_id: str element1_type: str element1_name: str element2_id: str element2_type: str element2_name: str clash_type: str distance: float location: Tuple[float, float, float] severity: str class ClashDetector: """Detect clashes between BIM elements""" def __init__(self, ifc_path: str): self.model = ifcopenshell.open(ifc_path) self.settings = ifcopenshell.geom.settings() self.settings.set(self.settings.USE_WORLD_COORDS, True) self._geometry_cache = {} self.clashes: List[Clash] = [] def _get_geometry(self, element): """Get or compute element geometry""" if element.GlobalId in self._geometry_cache: return self._geometry_cache[element.GlobalId] try: shape = ifcopenshell.geom.create_shape(self.settings, element) verts = np.array(shape.geometry.verts).reshape(-1, 3) faces = np.array(shape.geometry.faces).reshape(-1, 3) geom = { 'vertices': verts, 'faces': faces, 'min': verts.min(axis=0), 'max': verts.max(axis=0), 'center': verts.mean(axis=0) } self._geometry_cache[element.GlobalId] = geom return geom except: return None def detect_hard_clashes(self, group1_types: List[str], group2_types: List[str]) -> List[Clash]: """Detect hard clashes (physical intersections) between two groups""" group1 = [] for ifc_type in group1_types: group1.extend(self.model.by_type(ifc_type)) group2 = [] for ifc_type in group2_types: group2.extend(self.model.by_type(ifc_type)) clashes = [] for elem1 in group1: geom1 = self._get_geometry(elem1) if geom1 is None: continue for elem2 in group2: if elem1.GlobalId == elem2.GlobalId: continue geom2 = self._get_geometry(elem2) if geom2 is None: continue # Bounding box check (fast filter) if not self._bbox_intersect(geom1, geom2): continue # Detailed check intersection = self._check_intersection(geom1, geom2) if intersection['intersects']: clash = Clash( element1_id=elem1.GlobalId, element1_type=elem1.is_a(), element1_name=elem1.Name or '', element2_id=elem2.GlobalId, element2_type=elem2.is_a(), element2_name=elem2.Name or '', clash_type='Hard', distance=intersection['distance'], location=tuple(intersection['point']), severity=self._classify_severity(intersection['distance']) ) clashes.append(clash) self.clashes.extend(clashes) return clashes def detect_soft_clashes(self, group1_types: List[str], group2_types: List[str], clearance: float = 0.1) -> List[Clash]: """Detect soft clashes (clearance violations)""" group1 = [] for ifc_type in group1_types: group1.extend(self.model.by_type(ifc_type)) group2 = [] for ifc_type in group2_types: group2.extend(self.model.by_type(ifc_type)) clashes = [] for elem1 in group1: geom1 = self._get_geometry(elem1) if geom1 is None: continue for elem2 in group2: if elem1.GlobalId == elem2.GlobalId: continue geom2 = self._get_geometry(elem2) if geom2 is None: continue # Check if within clearance distance distance = self._min_distance(geom1, geom2) if distance < clearance and distance > 0: clash = Clash( element1_id=elem1.GlobalId, element1_type=elem1.is_a(), element1_name=elem1.Name or '', element2_id=elem2.GlobalId, element2_type=elem2.is_a(), element2_name=elem2.Name or '', clash_type='Soft', distance=distance, location=tuple((geom1['center'] + geom2['center']) / 2), severity='Medium' if distance < clearance/2 else 'Low' ) clashes.append(clash) self.clashes.extend(clashes) return clashes def _bbox_intersect(self, geom1: Dict, geom2: Dict) -> bool: """Check if bounding boxes intersect""" return (np.all(geom1['max'] >= geom2['min']) and np.all(geom2['max'] >= geom1['min'])) def _check_intersection(self, geom1: Dict, geom2: Dict) -> Dict: """Check for actual geometry intersection""" # Simplified check using closest points tree1 = cKDTree(geom1['vertices']) distances, _ = tree1.query(geom2['vertices'], k=1) min_dist = distances.min() if min_dist < 0.001: # Intersection threshold intersection_idx = np.argmin(distances) return { 'intersects': True, 'distance': min_dist, 'point': geom2['vertices'][intersection_idx] } return {'intersects': False, 'distance': min_dist, 'point': None} def _min_distance(self, geom1: Dict, geom2: Dict) -> float: """Calculate minimum distance between geometries""" tree1 = cKDTree(geom1['vertices']) distances, _ = tree1.query(geom2['vertices'], k=1) return distances.min() def _classify_severity(self, distance: float) -> str: """Classify clash severity""" if distance < 0.01: return 'Critical' elif distance < 0.05: return 'High' elif distance < 0.1: return 'Medium' else: return 'Low' def get_clash_report(self) -> pd.DataFrame: """Generate clash report as DataFrame""" if not self.clashes: return pd.DataFrame() return pd.DataFrame([ { 'Element1_ID': c.element1_id, 'Element1_Type': c.element1_type, 'Element1_Name': c.element1_name, 'Element2_ID': c.element2_id, 'Element2_Type': c.element2_type, 'Element2_Name': c.element2_name, 'Clash_Type': c.clash_type, 'Distance_m': c.distance, 'Location_X': c.location[0], 'Location_Y': c.location[1], 'Location_Z': c.location[2], 'Severity': c.severity } for c in self.clashes ]) def get_summary(self) -> Dict: """Get clash detection summary""" df = self.get_clash_report() if df.empty: return {'total': 0} return { 'total': len(self.clashes), 'by_type': df['Clash_Type'].value_counts().to_dict(), 'by_severity': df['Severity'].value_counts().to_dict(), 'critical_count': len(df[df['Severity'] == 'Critical']), 'element_types_involved': df['Element1_Type'].unique().tolist() + df['Element2_Type'].unique().tolist() }
Clash Sets Configuration
Common Clash Test Sets
# Define common clash test configurations CLASH_SETS = { 'structure_vs_mep': { 'group1': ['IfcColumn', 'IfcBeam', 'IfcWall', 'IfcSlab'], 'group2': ['IfcPipeSegment', 'IfcDuctSegment', 'IfcCableSegment'], 'clearance': 0.05, 'description': 'Structural elements vs MEP systems' }, 'piping_vs_hvac': { 'group1': ['IfcPipeSegment', 'IfcPipeFitting'], 'group2': ['IfcDuctSegment', 'IfcDuctFitting'], 'clearance': 0.10, 'description': 'Plumbing vs HVAC conflicts' }, 'doors_clearance': { 'group1': ['IfcDoor'], 'group2': ['IfcColumn', 'IfcWall'], 'clearance': 0.90, # Door swing clearance 'description': 'Door opening clearances' }, 'electrical_vs_plumbing': { 'group1': ['IfcCableSegment', 'IfcElectricDistributionBoard'], 'group2': ['IfcPipeSegment', 'IfcSanitaryTerminal'], 'clearance': 0.15, 'description': 'Electrical safety clearance from water' }, 'ceiling_vs_mep': { 'group1': ['IfcCovering'], 'group2': ['IfcPipeSegment', 'IfcDuctSegment', 'IfcCableCarrierSegment'], 'clearance': 0.05, 'description': 'Ceiling clearance for MEP' } } def run_all_clash_tests(detector: ClashDetector, clash_sets: Dict = None) -> Dict: """Run all configured clash tests""" if clash_sets is None: clash_sets = CLASH_SETS results = {} for test_name, config in clash_sets.items(): print(f"Running: {config['description']}...") # Hard clashes hard = detector.detect_hard_clashes(config['group1'], config['group2']) # Soft clashes soft = detector.detect_soft_clashes( config['group1'], config['group2'], config['clearance'] ) results[test_name] = { 'description': config['description'], 'hard_clashes': len(hard), 'soft_clashes': len(soft), 'total': len(hard) + len(soft) } return results
Report Generation
Export Clash Report
def export_clash_report(detector: ClashDetector, output_path: str): """Export comprehensive clash report to Excel""" df = detector.get_clash_report() summary = detector.get_summary() with pd.ExcelWriter(output_path, engine='openpyxl') as writer: # Summary sheet summary_df = pd.DataFrame([{ 'Total Clashes': summary['total'], 'Critical': summary.get('critical_count', 0), 'Hard Clashes': summary['by_type'].get('Hard', 0), 'Soft Clashes': summary['by_type'].get('Soft', 0) }]) summary_df.to_excel(writer, sheet_name='Summary', index=False) # All clashes if not df.empty: df.to_excel(writer, sheet_name='All_Clashes', index=False) # By severity for severity in ['Critical', 'High', 'Medium', 'Low']: severity_df = df[df['Severity'] == severity] if not severity_df.empty: severity_df.to_excel(writer, sheet_name=severity, index=False) return output_path def generate_clash_html_report(detector: ClashDetector, output_path: str): """Generate HTML report with visualizations""" df = detector.get_clash_report() summary = detector.get_summary() html = f""" <!DOCTYPE html> <html> <head> <title>Clash Detection Report</title> <style> body {{ font-family: Arial, sans-serif; margin: 20px; }} .summary {{ background: #f5f5f5; padding: 20px; border-radius: 8px; }} .critical {{ background: #ffebee; }} .high {{ background: #fff3e0; }} table {{ border-collapse: collapse; width: 100%; margin-top: 20px; }} th, td {{ border: 1px solid #ddd; padding: 8px; text-align: left; }} th {{ background: #4CAF50; color: white; }} </style> </head> <body> <h1>Clash Detection Report</h1> <div class="summary"> <h2>Summary</h2> <p><strong>Total Clashes:</strong> {summary['total']}</p> <p><strong>Critical:</strong> {summary.get('critical_count', 0)}</p> </div> <h2>Clash Details</h2> {df.to_html(index=False) if not df.empty else '<p>No clashes found</p>'} </body> </html> """ with open(output_path, 'w') as f: f.write(html) return output_path
Quick Reference
| Clash Type | Description | Typical Clearance |
|---|---|---|
| Hard Clash | Physical intersection | 0 mm |
| Soft Clash | Clearance violation | 50-150 mm |
| Workflow | Schedule conflict | N/A |
| Severity | Distance | Action Required |
|---|---|---|
| Critical | < 10 mm | Immediate redesign |
| High | 10-50 mm | Priority fix |
| Medium | 50-100 mm | Review needed |
| Low | > 100 mm | Monitor |
Resources
- IfcOpenShell: https://ifcopenshell.org
- DDC Website: https://datadrivenconstruction.io
Next Steps
- See
for time-based clash analysis4d-simulation - See
for validation workflowsbim-validation-pipeline - See
for element dataifc-data-extraction