You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
436 lines
10 KiB
436 lines
10 KiB
2 years ago
|
//Docs: https://agviegas.github.io/ifcjs-docs/#/
|
||
|
|
||
|
import * as WebIFC from './ifc/web-ifc-api.js';
|
||
|
import {
|
||
|
FileLoader,
|
||
|
Loader,
|
||
|
Mesh,
|
||
|
Color,
|
||
|
MeshBasicMaterial,
|
||
|
MeshLambertMaterial,
|
||
|
DoubleSide,
|
||
|
Matrix4,
|
||
|
BufferGeometry,
|
||
|
BufferAttribute,
|
||
|
} from '../../../build/three.module.js';
|
||
|
import {BufferGeometryUtils} from "../utils/BufferGeometryUtils.js"
|
||
|
|
||
|
const ifcAPI = new WebIFC.IfcAPI();
|
||
|
|
||
|
class IFCLoader extends Loader {
|
||
|
|
||
|
constructor( manager ) {
|
||
|
|
||
|
super( manager );
|
||
|
this.modelID = 0;
|
||
|
this.mapFaceindexID = {};
|
||
|
this.mapIDGeometry = {};
|
||
|
this.selectedObjects = [];
|
||
|
this.highlightMaterial = new MeshBasicMaterial({ color: 0xff0000, depthTest: false, side: DoubleSide });
|
||
|
|
||
|
}
|
||
|
|
||
|
load( url, onLoad, onProgress, onError ) {
|
||
|
|
||
|
const scope = this;
|
||
|
|
||
|
const loader = new FileLoader( scope.manager );
|
||
|
loader.setPath( scope.path );
|
||
|
loader.setResponseType( 'arraybuffer' );
|
||
|
loader.setRequestHeader( scope.requestHeader );
|
||
|
loader.setWithCredentials( scope.withCredentials );
|
||
|
loader.load(
|
||
|
url,
|
||
|
async function ( buffer ) {
|
||
|
|
||
|
try {
|
||
|
|
||
|
onLoad( await scope.parse( buffer ) );
|
||
|
|
||
|
} catch ( e ) {
|
||
|
|
||
|
if ( onError ) {
|
||
|
|
||
|
onError( e );
|
||
|
|
||
|
} else {
|
||
|
|
||
|
console.error( e );
|
||
|
|
||
|
}
|
||
|
|
||
|
scope.manager.itemError( url );
|
||
|
|
||
|
}
|
||
|
|
||
|
},
|
||
|
onProgress,
|
||
|
onError
|
||
|
);
|
||
|
|
||
|
}
|
||
|
|
||
|
setWasmPath( path ) {
|
||
|
|
||
|
ifcAPI.SetWasmPath( path );
|
||
|
|
||
|
}
|
||
|
|
||
|
getExpressId( faceIndex ) {
|
||
|
|
||
|
for (let index in this.mapFaceindexID) {
|
||
|
|
||
|
if (parseInt(index) >= faceIndex) return this.mapFaceindexID[index];
|
||
|
|
||
|
}
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
highlightItems( expressIds, scene, material = this.highlightMaterial ) {
|
||
|
|
||
|
this.removePreviousSelection(scene);
|
||
|
|
||
|
expressIds.forEach((id) => {
|
||
|
|
||
|
if (!this.mapIDGeometry[id]) return;
|
||
|
var mesh = new Mesh(this.mapIDGeometry[id], material);
|
||
|
mesh.renderOrder = 1;
|
||
|
scene.add(mesh);
|
||
|
this.selectedObjects.push(mesh);
|
||
|
return;
|
||
|
|
||
|
});
|
||
|
}
|
||
|
|
||
|
removePreviousSelection( scene ) {
|
||
|
|
||
|
if (this.selectedObjects.length > 0){
|
||
|
|
||
|
this.selectedObjects.forEach((object) => scene.remove(object));
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
setItemsVisibility( expressIds, geometry, visible = false ) {
|
||
|
|
||
|
this.setupVisibility(geometry);
|
||
|
var previous = 0;
|
||
|
|
||
|
for (var current in this.mapFaceindexID) {
|
||
|
|
||
|
if (expressIds.includes(this.mapFaceindexID[current])) {
|
||
|
|
||
|
for (var i = previous; i <= current; i++) this.setVertexVisibility(geometry, i, visible);
|
||
|
|
||
|
}
|
||
|
|
||
|
previous = current;
|
||
|
|
||
|
}
|
||
|
|
||
|
geometry.attributes.visibility.needsUpdate = true;
|
||
|
|
||
|
}
|
||
|
|
||
|
setVertexVisibility( geometry, index, visible ) {
|
||
|
|
||
|
var isVisible = visible ? 0 : 1;
|
||
|
var geoIndex = geometry.index.array;
|
||
|
geometry.attributes.visibility.setX(geoIndex[3 * index], isVisible);
|
||
|
geometry.attributes.visibility.setX(geoIndex[3 * index + 1], isVisible);
|
||
|
geometry.attributes.visibility.setX(geoIndex[3 * index + 2], isVisible);
|
||
|
|
||
|
}
|
||
|
|
||
|
setupVisibility( geometry ) {
|
||
|
|
||
|
if (!geometry.attributes.visibility) {
|
||
|
|
||
|
var visible = new Float32Array(geometry.getAttribute('position').count);
|
||
|
geometry.setAttribute('visibility', new BufferAttribute(visible, 1));
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
getItemProperties( elementID, all = false ) {
|
||
|
|
||
|
const properties = ifcAPI.GetLine(this.modelID, elementID);
|
||
|
|
||
|
if (all) {
|
||
|
|
||
|
const propSetIds = this.getAllRelatedItemsOfType(elementID, WebIFC.IFCRELDEFINESBYPROPERTIES, "RelatedObjects", "RelatingPropertyDefinition");
|
||
|
properties.hasPropertySets = propSetIds.map((id) => ifcAPI.GetLine(this.modelID, id, true));
|
||
|
|
||
|
const typeId = this.getAllRelatedItemsOfType(elementID, WebIFC.IFCRELDEFINESBYTYPE, "RelatedObjects", "RelatingType");
|
||
|
properties.hasType = typeId.map((id) => ifcAPI.GetLine(this.modelID, id, true));
|
||
|
|
||
|
}
|
||
|
|
||
|
// properties.type = properties.constructor.name;
|
||
|
return properties;
|
||
|
|
||
|
}
|
||
|
|
||
|
getSpatialStructure() {
|
||
|
|
||
|
let lines = ifcAPI.GetLineIDsWithType(this.modelID, WebIFC.IFCPROJECT);
|
||
|
let ifcProjectId = lines.get(0);
|
||
|
let ifcProject = ifcAPI.GetLine(this.modelID, ifcProjectId);
|
||
|
this.getAllSpatialChildren(ifcProject);
|
||
|
return ifcProject;
|
||
|
|
||
|
}
|
||
|
|
||
|
getAllSpatialChildren( spatialElement ) {
|
||
|
|
||
|
const id = spatialElement.expressID;
|
||
|
const spatialChildrenID = this.getAllRelatedItemsOfType(id, WebIFC.IFCRELAGGREGATES, "RelatingObject", "RelatedObjects");
|
||
|
spatialElement.hasSpatialChildren = spatialChildrenID.map((id) => ifcAPI.GetLine(this.modelID, id, false));
|
||
|
spatialElement.hasChildren = this.getAllRelatedItemsOfType(id, WebIFC.IFCRELCONTAINEDINSPATIALSTRUCTURE, "RelatingStructure", "RelatedElements");
|
||
|
spatialElement.hasSpatialChildren.forEach(child => this.getAllSpatialChildren(child));
|
||
|
|
||
|
}
|
||
|
|
||
|
getAllRelatedItemsOfType ( elementID, type, relation, relatedProperty ) {
|
||
|
|
||
|
const lines = ifcAPI.GetLineIDsWithType(this.modelID, type);
|
||
|
const IDs = [];
|
||
|
|
||
|
for (let i = 0; i < lines.size(); i++) {
|
||
|
|
||
|
const relID = lines.get(i);
|
||
|
const rel = ifcAPI.GetLine(this.modelID, relID);
|
||
|
const relatedItems = rel[relation];
|
||
|
let foundElement = false;
|
||
|
|
||
|
if (Array.isArray(relatedItems)){
|
||
|
|
||
|
relatedItems.forEach((relID) => {
|
||
|
|
||
|
if (relID.value === elementID) foundElement = true;
|
||
|
|
||
|
});
|
||
|
}
|
||
|
else foundElement = (relatedItems.value === elementID);
|
||
|
|
||
|
if (foundElement) {
|
||
|
|
||
|
var element = rel[relatedProperty];
|
||
|
if (!Array.isArray(element)) IDs.push(element.value);
|
||
|
else element.forEach(ele => IDs.push(ele.value))
|
||
|
|
||
|
}
|
||
|
}
|
||
|
return IDs;
|
||
|
}
|
||
|
|
||
|
async parse( buffer ) {
|
||
|
|
||
|
const geometryByMaterials = {};
|
||
|
const mapIDGeometry = this.mapIDGeometry;
|
||
|
const mapFaceindexID = this.mapFaceindexID;
|
||
|
|
||
|
if ( ifcAPI.wasmModule === undefined ) {
|
||
|
|
||
|
await ifcAPI.Init();
|
||
|
|
||
|
}
|
||
|
|
||
|
const data = new Uint8Array( buffer );
|
||
|
this.modelID = ifcAPI.OpenModel( 'example.ifc', data );
|
||
|
return loadAllGeometry( this.modelID );
|
||
|
|
||
|
function loadAllGeometry(modelID) {
|
||
|
|
||
|
saveAllPlacedGeometriesByMaterial(modelID);
|
||
|
return generateAllGeometriesByMaterial();
|
||
|
|
||
|
}
|
||
|
|
||
|
function generateAllGeometriesByMaterial() {
|
||
|
|
||
|
const { materials, geometries } = getMaterialsAndGeometries();
|
||
|
const allGeometry = BufferGeometryUtils.mergeBufferGeometries(geometries, true);
|
||
|
return new Mesh(allGeometry, materials);
|
||
|
|
||
|
}
|
||
|
|
||
|
function getMaterialsAndGeometries() {
|
||
|
|
||
|
const materials = [];
|
||
|
const geometries = [];
|
||
|
let totalFaceCount = 0;
|
||
|
|
||
|
for (let i in geometryByMaterials) {
|
||
|
|
||
|
materials.push(geometryByMaterials[i].material);
|
||
|
const currentGeometries = geometryByMaterials[i].geometry;
|
||
|
geometries.push(BufferGeometryUtils.mergeBufferGeometries(currentGeometries));
|
||
|
|
||
|
for (let j in geometryByMaterials[i].indices) {
|
||
|
|
||
|
const globalIndex = parseInt(j, 10) + parseInt(totalFaceCount, 10);
|
||
|
mapFaceindexID[globalIndex] = geometryByMaterials[i].indices[j];
|
||
|
|
||
|
}
|
||
|
|
||
|
totalFaceCount += geometryByMaterials[i].lastIndex;
|
||
|
|
||
|
}
|
||
|
|
||
|
return { materials, geometries };
|
||
|
|
||
|
}
|
||
|
|
||
|
function saveAllPlacedGeometriesByMaterial(modelID) {
|
||
|
|
||
|
const flatMeshes = ifcAPI.LoadAllGeometry(modelID);
|
||
|
|
||
|
for (let i = 0; i < flatMeshes.size(); i++) {
|
||
|
|
||
|
const flatMesh = flatMeshes.get(i);
|
||
|
const productId = flatMesh.expressID;
|
||
|
const placedGeometries = flatMesh.geometries;
|
||
|
|
||
|
for (let j = 0; j < placedGeometries.size(); j++) {
|
||
|
|
||
|
savePlacedGeometryByMaterial(modelID, placedGeometries.get(j), productId);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function savePlacedGeometryByMaterial(modelID, placedGeometry, productId) {
|
||
|
|
||
|
const geometry = getBufferGeometry(modelID, placedGeometry);
|
||
|
geometry.computeVertexNormals();
|
||
|
const matrix = getMeshMatrix(placedGeometry.flatTransformation);
|
||
|
geometry.applyMatrix4(matrix);
|
||
|
storeGeometryForHighlight(productId, geometry);
|
||
|
saveGeometryByMaterial(geometry, placedGeometry, productId);
|
||
|
|
||
|
}
|
||
|
|
||
|
function getBufferGeometry(modelID, placedGeometry) {
|
||
|
|
||
|
const geometry = ifcAPI.GetGeometry(modelID, placedGeometry.geometryExpressID);
|
||
|
const verts = ifcAPI.GetVertexArray(geometry.GetVertexData(), geometry.GetVertexDataSize());
|
||
|
const indices = ifcAPI.GetIndexArray(geometry.GetIndexData(), geometry.GetIndexDataSize());
|
||
|
return ifcGeometryToBuffer(verts, indices);
|
||
|
|
||
|
}
|
||
|
|
||
|
function getMeshMatrix(matrix) {
|
||
|
|
||
|
const mat = new Matrix4();
|
||
|
mat.fromArray(matrix);
|
||
|
return mat;
|
||
|
|
||
|
}
|
||
|
|
||
|
function storeGeometryForHighlight(productId, geometry) {
|
||
|
|
||
|
if (!mapIDGeometry[productId]) {
|
||
|
|
||
|
mapIDGeometry[productId] = geometry;
|
||
|
return;
|
||
|
|
||
|
}
|
||
|
|
||
|
const geometries = [mapIDGeometry[productId], geometry];
|
||
|
mapIDGeometry[productId] = BufferGeometryUtils.mergeBufferGeometries(geometries, true);
|
||
|
|
||
|
}
|
||
|
|
||
|
function ifcGeometryToBuffer(vertexData, indexData) {
|
||
|
|
||
|
const geometry = new BufferGeometry();
|
||
|
const { vertices, normals } = extractVertexData(vertexData);
|
||
|
geometry.setAttribute('position', new BufferAttribute(new Float32Array(vertices), 3));
|
||
|
geometry.setAttribute('normal', new BufferAttribute(new Float32Array(normals), 3));
|
||
|
geometry.setIndex(new BufferAttribute(indexData, 1));
|
||
|
return geometry;
|
||
|
|
||
|
}
|
||
|
|
||
|
function extractVertexData(vertexData) {
|
||
|
|
||
|
const vertices = [];
|
||
|
const normals = [];
|
||
|
let isNormalData = false;
|
||
|
|
||
|
for (let i = 0; i < vertexData.length; i++) {
|
||
|
|
||
|
isNormalData ? normals.push(vertexData[i]) : vertices.push(vertexData[i]);
|
||
|
if ((i + 1) % 3 == 0) isNormalData = !isNormalData;
|
||
|
|
||
|
}
|
||
|
|
||
|
return { vertices, normals };
|
||
|
|
||
|
}
|
||
|
|
||
|
function saveGeometryByMaterial(geometry, placedGeometry, productId) {
|
||
|
|
||
|
const color = placedGeometry.color;
|
||
|
const id = `${color.x}${color.y}${color.z}${color.w}`;
|
||
|
createMaterial(id, color);
|
||
|
const currentGeometry = geometryByMaterials[id];
|
||
|
currentGeometry.geometry.push(geometry);
|
||
|
currentGeometry.lastIndex += geometry.index.count / 3;
|
||
|
currentGeometry.indices[currentGeometry.lastIndex] = productId;
|
||
|
|
||
|
}
|
||
|
|
||
|
function createMaterial(id, color) {
|
||
|
|
||
|
if (!geometryByMaterials[id]){
|
||
|
|
||
|
const col = new Color(color.x, color.y, color.z);
|
||
|
const newMaterial = new MeshLambertMaterial({ color: col, side: DoubleSide });
|
||
|
newMaterial.onBeforeCompile = materialHider;
|
||
|
newMaterial.transparent = color.w !== 1;
|
||
|
if (newMaterial.transparent) newMaterial.opacity = color.w;
|
||
|
geometryByMaterials[id] = initializeGeometryByMaterial(newMaterial);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function initializeGeometryByMaterial(newMaterial) {
|
||
|
|
||
|
return {
|
||
|
material: newMaterial,
|
||
|
geometry: [],
|
||
|
indices: {},
|
||
|
lastIndex: 0
|
||
|
|
||
|
};
|
||
|
}
|
||
|
|
||
|
function materialHider(shader) {
|
||
|
shader.vertexShader = `
|
||
|
attribute float sizes;
|
||
|
attribute float visibility;
|
||
|
varying float vVisible;
|
||
|
${shader.vertexShader}`.replace(
|
||
|
`#include <fog_vertex>`,
|
||
|
`#include <fog_vertex>
|
||
|
vVisible = visibility;
|
||
|
`
|
||
|
);
|
||
|
shader.fragmentShader = `
|
||
|
varying float vVisible;
|
||
|
${shader.fragmentShader}`.replace(
|
||
|
`#include <clipping_planes_fragment>`,
|
||
|
`
|
||
|
if (vVisible > 0.5) discard;
|
||
|
#include <clipping_planes_fragment>`
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
export { IFCLoader };
|