@giro3d/piero 1.0.0-beta.2 → 1.0.0-beta.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/dist/Components.cjs.js +3 -2
  2. package/dist/Components.cjs.js.map +1 -1
  3. package/dist/Components.es.js +3193 -1778
  4. package/dist/Components.es.js.map +1 -1
  5. package/dist/assets/piero.css +2 -2
  6. package/dist/index.cjs.js +7 -7
  7. package/dist/index.cjs.js.map +1 -1
  8. package/dist/index.es.js +3192 -4319
  9. package/dist/index.es.js.map +1 -1
  10. package/dist/modules.cjs.js +1 -2
  11. package/dist/modules.cjs.js.map +1 -1
  12. package/dist/modules.es.js +981 -965
  13. package/dist/modules.es.js.map +1 -1
  14. package/dist/src/api/SearchApi.d.ts +34 -0
  15. package/dist/src/api/ViewApi.d.ts +2 -0
  16. package/dist/src/api/WidgetApi.d.ts +17 -0
  17. package/dist/src/api/index.d.ts +2 -1
  18. package/dist/src/components/CoordinateInput.vue.d.ts +31 -0
  19. package/dist/src/components/SearchOverlay.vue.d.ts +4 -5
  20. package/dist/src/context.d.ts +10 -0
  21. package/dist/src/events.d.ts +7 -1
  22. package/dist/src/modules/OpenLayersMinimap.d.ts +7 -0
  23. package/dist/src/modules/PostProcessEntities.d.ts +16 -0
  24. package/dist/src/modules/crossSectionAnalysis/CrossSection.vue.d.ts +1 -1
  25. package/dist/src/modules/crossSectionAnalysis/CrossSectionHelper.d.ts +4 -0
  26. package/dist/src/modules/crossSectionAnalysis/CrossSectionManager.d.ts +7 -0
  27. package/dist/src/modules/crossSectionAnalysis/store.d.ts +22 -2
  28. package/dist/src/modules/geocoding/CoordinatesSearch.d.ts +7 -0
  29. package/dist/src/modules/geocoding/FrenchBanGeocoder.d.ts +10 -0
  30. package/dist/src/modules/index.d.ts +5 -1
  31. package/dist/src/modules/minimap/OpenLayersMinimapComponent.vue.d.ts +8 -0
  32. package/dist/src/services/CameraController.d.ts +3 -3
  33. package/dist/src/services/Giro3DManager.d.ts +2 -0
  34. package/dist/src/services/SceneCursorManager.d.ts +19 -0
  35. package/dist/src/stores/giro3d.d.ts +1 -7
  36. package/dist/src/stores/search.d.ts +12 -0
  37. package/dist/src/stores/widgets.d.ts +12 -0
  38. package/dist/src/utils/Types.d.ts +1 -0
  39. package/package.json +2 -2
  40. package/dist/src/components/MinimapView.vue.d.ts +0 -4
  41. package/dist/src/providers/BanProvider.d.ts +0 -11
  42. package/dist/src/providers/Geocoding.d.ts +0 -7
  43. package/dist/src/services/MinimapController.d.ts +0 -19
@@ -1,3 +1,2 @@
1
- "use strict";var W=Object.defineProperty;var X=(c,t,e)=>t in c?W(c,t,{enumerable:!0,configurable:!0,writable:!0,value:e}):c[t]=e;var y=(c,t,e)=>X(c,typeof t!="symbol"?t+"":t,e);Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const p=require("./Components.cjs.js"),r=require("vue"),U=require("pinia"),Z=require("@giro3d/giro3d/core/geographic/Coordinates"),R=require("@giro3d/giro3d/entities/Entity3D"),Q=require("@giro3d/giro3d/gui/EntityPanel"),T=require("openbim-components"),J=require("@giro3d/giro3d/gui/EntityInspector"),K=require("@giro3d/giro3d/entities/PointCloud"),ee=require("@giro3d/giro3d/sources/PotreeSource"),F=require("shepherd.js"),Y=U.defineStore("crossSection",()=>{const c=r.ref(0),t=r.ref(new p.Vector3(0,0,0)),e=r.ref(!1);function o(n){e.value=n}function s(n){c.value=n}function i(n){t.value=n}return{center:t,enable:e,orientation:c,setCenter:i,setEnabled:o,setOrientation:s}}),te={class:"input-group mb-3"},ne={class:"input-group mb-3"},oe=["value"],se={class:"input-group mb-3"},ie=["value"],re={class:"input-group"},ae={class:"input-group"},le=["value"],ce=["value"],ue=r.defineComponent({__name:"CrossSection",setup(c){const t=Y();function e(i){t.setOrientation(i)}function o(i){const n=t.center.clone();n.setX(i),t.setCenter(n)}function s(i){const n=t.center.clone();n.setY(i),t.setCenter(n)}return(i,n)=>(r.openBlock(),r.createElementBlock("div",null,[r.createElementVNode("div",te,[r.createVNode(p._sfc_main,{"model-value":r.unref(t).enable,"onUpdate:modelValue":n[0]||(n[0]=l=>r.unref(t).setEnabled(l)),id:"cross-section-enable",title:"foo"},null,8,["model-value"]),n[5]||(n[5]=r.createElementVNode("label",{for:"cross-section-enable",class:"form-label"},"Enable cross section",-1))]),r.createElementVNode("div",ne,[n[6]||(n[6]=r.createElementVNode("label",{for:"plane-orientation-range",class:"form-label"},"Plane orientation (0-360°)",-1)),r.createElementVNode("input",{id:"plane-orientation-range",title:"Altitude",type:"range",class:"form-range",min:"0",step:"0.1",max:"360",value:r.unref(t).orientation,onInput:n[1]||(n[1]=l=>e(Number.parseFloat(l.target.value)))},null,40,oe)]),r.createElementVNode("div",se,[r.createElementVNode("input",{type:"number",class:"form-control",id:"plane-orientation-number",step:"0.1",value:r.unref(t).orientation,onInput:n[2]||(n[2]=l=>e(Number.parseFloat(l.target.value)))},null,40,ie)]),n[8]||(n[8]=r.createElementVNode("hr",null,null,-1)),r.createElementVNode("div",re,[n[7]||(n[7]=r.createElementVNode("label",{class:"form-label"},"Center (x, y)",-1)),r.createElementVNode("div",ae,[r.createElementVNode("input",{type:"number",class:"form-control",id:"plane-center-x",value:r.unref(t).center.x,onInput:n[3]||(n[3]=l=>o(Number.parseFloat(l.target.value)))},null,40,le),r.createElementVNode("input",{type:"number",class:"form-control",id:"plane-center-y",value:r.unref(t).center.y,onInput:n[4]||(n[4]=l=>s(Number.parseFloat(l.target.value)))},null,40,ce)])])]))}});class de{constructor(t){y(this,"_store",Y());this.context=t,t.events.addEventListener("ready",()=>{const e=t.configuration.default_crs,o=t.configuration.analysis.cross_section;this._store.setOrientation(o.orientation);const s=o.pivot,i=new Z(s.crs??e,s.x,s.y,0).as(e);this._store.setCenter(i.toVector3()),this._store.$onAction(({after:n,name:l})=>{n(()=>{switch(l){case"setCenter":case"setEnabled":case"setOrientation":this.updateCrossSection();break}})}),this.updateCrossSection()})}dispose(){}updateCrossSection(){const t=[];if(this._store.enable){const o=p.MathUtils.DEG2RAD*this._store.orientation,s=Math.cos(o),i=Math.sin(o),n=new p.Vector3(s,i,0),l=new p.Plane(n,0).distanceToPoint(this._store.center),b=new p.Plane(n,-l);t.push(b)}const e=this.context.view.getInstance();e.renderer.clippingPlanes=t;for(const o of this.context.view.getInstance().getEntities())R.isEntity3D(o)&&o.dispatchEvent({clippingPlanes:t,type:"clippingPlanes-property-changed"});e.notifyChange()}}class pe{constructor(){y(this,"id","builtin-cross-section-analysis");y(this,"name","Cross section");y(this,"_manager",null)}initialize(t){t.analysis.registerTool({component:ue,icon:"bi-circle-half",name:"Cross section"}),this._manager=new de(t)}}function z(c){if("url"in c.config&&typeof c.config.url=="string")return c.config.url;if("source"in c.config&&typeof c.config.source=="object"&&"url"in c.config.source&&typeof c.config.source.url=="string")return c.config.source.url}function he(c){return z(c)!=null}class fe{constructor(){y(this,"id","builtin-download-dataset");y(this,"name","Download dataset");y(this,"_context",null)}initialize(t){this._context=t,t.datasets.registerDatasetAction({action:this.download.bind(this),icon:"bi-download",predicate:he,title:"Download dataset"})}download(t){const e=z(t);if(e==null){console.warn("invalid");return}const o=this._context,s=new URL(e).pathname.split("/"),i=s[s.length-1];console.info(`download ${z(t)}`),p.Fetcher.fetch(e).then(n=>{if(!n.ok)throw new Error(`${n.status} ${n.statusText}`);return n.blob()}).then(n=>{p.Download.downloadBlob(n,i),o.notifications.pushNotification({level:"success",text:i,title:"Download successful"})}).catch(n=>{console.error(n);const l=n instanceof Error?n.message:"Download failed";o.notifications.pushNotification({level:"error",text:l,title:t.name})})}}const $=U.defineStore("floodingPlane",()=>{const c=r.ref(0),t=r.ref(!1);function e(){return c.value}function o(i){c.value=i}function s(i){t.value=i}return{enable:t,getHeight:e,setEnabled:s,setHeight:o}}),me={class:"input-group mb-3"},ge={class:"input-group mb-3"},ye=["value"],be={class:"input-group mb-3"},_e=["value"],we=r.defineComponent({__name:"FloodingPlane",setup(c){const t=$();function e(o){t.setHeight(o)}return(o,s)=>(r.openBlock(),r.createElementBlock(r.Fragment,null,[r.createElementVNode("div",me,[r.createVNode(p._sfc_main,{"model-value":r.unref(t).enable,"onUpdate:modelValue":s[0]||(s[0]=i=>r.unref(t).setEnabled(i)),id:"flooding-plane-enable",title:"foo"},null,8,["model-value"]),s[3]||(s[3]=r.createElementVNode("label",{for:"flooding-plane-enable",class:"form-label"},"Enable flooding plane",-1))]),r.createElementVNode("div",ge,[s[5]||(s[5]=r.createElementVNode("label",{for:"flooding-altitude-range",class:"form-label"},"Altitude",-1)),r.createElementVNode("input",{id:"flooding-altitude-range",title:"Altitude",type:"range",class:"form-range",min:"0",step:"0.1",max:"500",value:r.unref(t).getHeight(),onInput:s[1]||(s[1]=i=>e(Number.parseFloat(i.target.value)))},null,40,ye),r.createElementVNode("div",be,[r.createElementVNode("input",{type:"number",class:"form-control",id:"flooding-altitude-number",step:"0.1",value:r.unref(t).getHeight(),onInput:s[2]||(s[2]=i=>e(Number.parseFloat(i.target.value)))},null,40,_e),s[4]||(s[4]=r.createElementVNode("span",{class:"input-group-text"},"m",-1))])])],64))}});class ve{constructor(){y(this,"geometry");y(this,"material");y(this,"object3D");y(this,"_height");this.geometry=new p.PlaneGeometry(1,1,1,1),this.material=new p.MeshBasicMaterial({color:43690,opacity:.5,transparent:!0}),this.object3D=new p.Mesh(this.geometry,this.material),this.object3D.renderOrder=2,this.visible=!1,this._height=0}set height(t){this._height=t}get height(){return this._height}set visible(t){this.object3D.visible=t}get visible(){return this.object3D.visible}dispose(){this.object3D.removeFromParent(),this.geometry.dispose(),this.material.dispose()}setPosition(t,e,o,s,i){this.object3D.scale.set(s,i,1),this.object3D.position.set(t,e,o),this.object3D.updateMatrixWorld()}}class xe{constructor(t){y(this,"_context");y(this,"_plane");y(this,"_store",$());this._context=t,this._plane=null,t.events.addEventListener("ready",()=>{this._store.$onAction(({after:e,name:o})=>{e(()=>{switch(o){case"setEnabled":this.updatePlane();break;case"setHeight":this.updatePlane();break}})})})}dispose(){this._plane&&(this._context.view.getInstance().remove(this._plane.object3D),this._plane.dispose())}async updatePlane(){this._plane||(this._plane=new ve,await this._context.view.getInstance().add(this._plane.object3D));const t=this._context.view.getBoundingBox(),e=t.getCenter(new p.Vector3),o=t.getSize(new p.Vector3);this._plane.visible=this._store.enable,this._plane.setPosition(e.x,e.y,this._store.getHeight(),o.x,o.y),this._context.view.getInstance().notifyChange()}}class Ce{constructor(){y(this,"id","builtin-flooding-plane-analysis");y(this,"name","Flooding plane");y(this,"_manager",null)}initialize(t){t.analysis.registerTool({component:we,icon:"bi-layers-half",name:"Flooding plane"}),this._manager=new xe(t)}}const D=new p.Matrix4,Ie={bbox:new p.MeshBasicMaterial({color:"#FFFF00",depthTest:!0,opacity:.2,transparent:!0}),selection:new p.MeshBasicMaterial({color:"#FF0000",depthTest:!1,opacity:.6,transparent:!0})},Pe=4186316022,Se=781010003,Ee=2655215786,Te=3242617779,Be=919958153,Ne=1307041759,G=1451395588,q=1883228015,Fe=[G,q],ke=[Pe,Se,Ee,Te,Be,Ne],H=c=>p.isObject(c)&&c.isIFCPickResult,A=class A extends R{constructor(e){super(new p.Group);y(this,"isIfcEntity",!0);y(this,"isPickableFeatures",!0);y(this,"type","IfcEntity");y(this,"_classificationCache");y(this,"_components");y(this,"_fragmentBoundingBox");y(this,"_fragmentClassifier");y(this,"_fragmentManager");y(this,"_ifcSelection");y(this,"_indexMap");y(this,"_model");y(this,"_source");this._source=e,this._components=new T.Components,this._components.ui.enabled=!1,this._components.scene=new T.SimpleScene(this._components),this._components.renderer=new T.SimpleRenderer(this._components,document.createElement("div")),this._components.camera=new T.SimpleCamera(this._components),this._components.raycaster=new T.SimpleRaycaster(this._components),this._components.init(),this._ifcSelection={bbox:{},selection:{}},this._indexMap={},this._classificationCache=null,this._fragmentBoundingBox=null}clearHighlight(e="selection"){for(const o of Object.keys(this._ifcSelection[e])){const s=this._fragmentManager.list[o],i=s==null?void 0:s.fragments[e];i!=null&&i.mesh.removeFromParent()}this.notifyChange(this),this._ifcSelection[e]={}}getBoundingBoxById(e){this.clearHighlight("bbox"),this.highlightById(e,"bbox");const o=this._fragmentBoundingBox;if(o===null)throw new Error("Must call initClassification before getBoundingBoxById");const s=this._fragmentManager;o.reset();const i=this._ifcSelection.bbox;if(!Object.keys(i).length)return;for(const g of Object.keys(i)){const S=s.list[g].fragments.bbox;o.addMesh(S.mesh)}const n=o.get(),{y:l,z:b}=n.min,{y:x,z:d}=n.max;return n.min.y=-d,n.max.y=-b,n.min.z=l,n.max.z=x,n.translate(this._model.position),this.clearHighlight("bbox"),n}getClassification(){if(this._classificationCache===null)throw new Error("Must call initClassification before getClassification");return this._classificationCache}getProperties(e){const o=[],s=this._model.properties;if(!s)return[];for(const i of this._indexMap[e]){const n=s[i];if(n==null)continue;const{name:l}=T.IfcPropertiesUtils.getEntityName(s,i);if(l!==null){if(n.type===G){const b=T.IfcPropertiesUtils.getPsetProps(s,i);if(b!==null)for(const x of b){if(s[x]==null)continue;const g=this.getProperty(x);g!==null&&o.push({parentName:l,...g})}}else if(n.type===q){const b=T.IfcPropertiesUtils.getQsetQuantities(s,i);if(b!==null)for(const x of b){const{key:d}=T.IfcPropertiesUtils.getQuantityValue(s,x);if(d===null)continue;const g=this.getProperty(x);g!==null&&o.push({parentName:l,...g})}}}}return o}getProperty(e){const o=this._model.properties;if(o===void 0)return null;const{name:s}=T.IfcPropertiesUtils.getEntityName(o,e);if(s===null)return null;const{value:i}=T.IfcPropertiesUtils.getQuantityValue(o,e);return{name:s,value:i}}highlight(e,o,s){this._ifcSelection[e][o.uuid]=new Set;const i=parseInt(s,10);this._ifcSelection[e][o.uuid].add(s),this.addComposites(e,o,i),this.regenerate(e,o.uuid);const n=o.fragment.group;if(n){const l=n.data[i][0];for(let b=0;b<l.length;b++){const x=l[b],d=n.keyFragments[x],g=this._fragmentManager.list[d];d in this._ifcSelection[e]||(this._ifcSelection[e][d]=new Set),this._ifcSelection[e][d].add(s),this.addComposites(e,g.mesh,i),this.regenerate(e,d)}}this.notifyChange(this)}highlightById(e,o="selection"){for(const s of Object.keys(e)){s in this._ifcSelection[o]||(this._ifcSelection[o][s]=new Set);const i=this._fragmentManager.list[s],n=new Set;for(const l of e[s])this._ifcSelection[o][s].add(l),n.add(parseInt(l,10));for(const l of n)this.addComposites(o,i.mesh,l);this.regenerate(o,s)}this.notifyChange(this)}pick(e,o){return super.pick(e,o).map(s=>({...s,entity:this,features:s.features,isIFCPickResult:!0,object:s.object}))}pickFeaturesFrom(e){var s,i,n;const o=e.object;if(o.fragment!=null&&e.instanceId!=null&&e.face){const l=o.fragment.getVertexBlockID(o.geometry,e.face.a),b=(s=o.fragment.getItemID(e.instanceId,l))==null?void 0:s.replace(/\..*/,"");if(b&&((n=(i=o.fragment.group)==null?void 0:i.properties)==null?void 0:n[b])!=null){const d=o.fragment.group.properties[b],I=[{ifcProperties:this.getProperties(b),itemProperties:d}];return e.features=I,I}}return[]}async initClassification(){this._classificationCache=await this.regenerateClassification(["storeys","entities"]),this._classificationCache.length===0&&(this._classificationCache=await this.regenerateClassification(["entities"])),this._fragmentBoundingBox=await this._components.tools.get(T.FragmentBoundingBox)}async preprocess(){const e=await p.Fetcher.fetchArrayBuffer(this._source.url);this._fragmentManager=await this._components.tools.get(T.FragmentManager),this._fragmentClassifier=await this._components.tools.get(T.FragmentClassifier);const o=new T.FragmentIfcLoader(this._components);o.settings.webIfc.COORDINATE_TO_ORIGIN=!0,o.settings.webIfc.OPTIMIZE_PROFILES=!0;const s=new Uint8Array(e);this._model=await o.load(s,this._source.name),this._model.rotateX(Math.PI/2);const i=new p.Vector3;if(this._source.at)this._source.at.toVector3(i),this._model.position.copy(i);else{const l=this._model.coordinationMatrix.clone().invert();i.applyMatrix4(l),this._model.position.set(i.x,-i.z,i.y)}this._model.updateWorldMatrix(!0,!0),this._model.updateMatrix(),this._model.updateMatrixWorld(!0),this.initializeEntityIndexes(),this._fragmentClassifier.byStorey(this._model),this._fragmentClassifier.byEntity(this._model),await this.initClassification(),this.object3d.add(this._model),this.onObjectCreated(this._model);const n=p.Fetcher.getContext(this._source.url);p.fillObject3DUserData(this,{filename:n.filename}),this.notifyChange(this.object3d)}addComposites(e,o,s){this.addHighlightToFragment(e,o.fragment);const i=o.fragment.composites[s];if(i)for(let n=1;n<i;n++){const l=T.toCompositeID(s,n);this._ifcSelection[e][o.uuid].add(l)}}addHighlightToFragment(e,o){if(!(e in o.fragments)){const s=o.addFragment(e,[Ie[e]]);o.blocks.count>1&&(s.setInstance(0,{ids:Array.from(o.ids),transform:D}),s.blocks.setVisibility(!1)),this._model.add(s.mesh),s.mesh.renderOrder=30,s.mesh.frustumCulled=!1,s.mesh.name=e,s.mesh.updateMatrixWorld(!0)}}initializeEntityIndexes(){this._indexMap={};const e=this._model.properties;if(e!==void 0)for(const o of ke)T.IfcPropertiesUtils.getRelationMap(e,o,(s,i)=>{const n=e[s];Fe.includes(n.type)||this.setEntityIndex(s);for(const l of i)this.setEntityIndex(l).add(s)})}regenerate(e,o){this.updateFragmentFill(e,o)}async regenerateClassification(e,o={}){const s=this._fragmentClassifier.get(),i=[],n=e[0],l=s[n];if(n==null||l==null)return i;for(const b of Object.keys(l)){const x={...o,[n]:[b]},d=await this._fragmentClassifier.find(x);if(Object.keys(d).length>0){const S=n[0].toUpperCase()+n.slice(1),P=await this.regenerateClassification(e.slice(1),x);i.push({children:P,fragments:d,name:b,treeItemName:S})}}return i}setEntityIndex(e){return e in this._indexMap||(this._indexMap[e]=new Set),this._indexMap[e]}updateFragmentFill(e,o){const s=this._ifcSelection[e][o],i=this._fragmentManager.list[o];if(i==null)return;const n=i.fragments[e];if(n==null)return;const l=i.mesh.parent;if(l==null)return;if(l.add(n.mesh),n.blocks.count>1)i.getInstance(0,D),n.setInstance(0,{ids:Array.from(i.ids),transform:D}),n.blocks.setVisibility(!0,s,!0);else{let x=0;for(const d of s){n.mesh.count=x+1;const{instanceID:g}=i.getInstanceAndBlockID(d);i.getInstance(g,D),n.setInstance(x,{ids:[d],transform:D}),x++}}}};y(A,"isIFCEntity",e=>p.isObject(e)&&e.isIfcEntity),y(A,"isIFCPickResult",e=>p.isObject(e)&&A.isIFCEntity(e.entity));let j=A;class Me extends J{constructor(t,e,o){super(t,e,o,{visibility:!0})}}const De={class:"d-flex"},Ae=["title"],Ve=["id"],Le={class:"list-unstyled border-start"},Oe=r.defineComponent({__name:"IfcSubtree",props:{classificationElement:{},ifcEntity:{}},setup(c){const t=c,e=p.MathUtils.generateUUID(),o=`#${e}`,s=r.ref(!1),i=p.useCameraStore(),l=p.useModuleStore().getModule(p.moduleId);function b(){if(l==null){console.warn("Cannot clip IFC element, ClippingBoxAnalysis module is not present");return}const g=t.ifcEntity.getBoundingBoxById(t.classificationElement.fragments);g&&!g.isEmpty()&&l.setClippingBox(g)}function x(){s.value=!0,t.ifcEntity.clearHighlight(),t.ifcEntity.highlightById(t.classificationElement.fragments),setTimeout(()=>s.value=!1,2e3)}function d(){const g=t.ifcEntity.getBoundingBoxById(t.classificationElement.fragments);g&&!g.isEmpty()&&i.lookTopDownAt(g)}return(g,I)=>{const S=r.resolveComponent("IfcSubtree",!0);return r.openBlock(),r.createElementBlock("div",null,[r.createElementVNode("div",De,[r.createElementVNode("span",{class:r.normalizeClass(["border rounded px-1 py-0 fw-normal",s.value?"text-danger border-danger":"text-secondary border-secondary"]),title:g.classificationElement.treeItemName},r.toDisplayString(g.classificationElement.treeItemName),11,Ae),g.classificationElement.children.length>0?(r.openBlock(),r.createBlock(p.IconList,{key:0},{default:r.withCtx(()=>[r.createVNode(p.IconListButton,{title:"Expand group",icon:"bi-chevron-down","data-bs-toggle":"collapse","data-bs-target":o,"aria-controls":r.unref(e),"aria-expanded":"true"},null,8,["aria-controls"])]),_:1})):r.createCommentVNode("",!0),r.createVNode(p._sfc_main$3,{class:r.normalizeClass(["label",s.value?"text-danger-emphasis":"text-muted"]),text:g.classificationElement.name,title:`Zoom to ${g.classificationElement.name}`,onClick:d},null,8,["class","text","title"]),r.createVNode(p.IconList,{class:"ms-1"},{default:r.withCtx(()=>[r.createVNode(p.IconListButton,{title:"Highlight",icon:"bi-highlighter",onClick:x}),r.unref(l)!=null?(r.openBlock(),r.createBlock(p.IconListButton,{key:0,title:"Clip to",icon:"bi-bounding-box",onClick:b})):r.createCommentVNode("",!0)]),_:1})]),g.classificationElement.children.length>0?(r.openBlock(),r.createElementBlock("div",{key:0,id:r.unref(e),class:"collapse show"},[r.createElementVNode("ul",Le,[(r.openBlock(!0),r.createElementBlock(r.Fragment,null,r.renderList(g.classificationElement.children,(P,B)=>(r.openBlock(),r.createElementBlock("li",{key:B},[r.createVNode(S,{"ifc-entity":t.ifcEntity,"classification-element":P},null,8,["ifc-entity","classification-element"])]))),128))])],8,Ve)):r.createCommentVNode("",!0)])}}}),ze=p._export_sfc(Oe,[["__scopeId","data-v-804ca598"]]),je={key:0},Re=r.defineComponent({__name:"IfcPropertyView",props:{dataset:{}},setup(c){const t=p.useDatasetStore(),e=c,o=p.refAndWatch(e.dataset,"isPreloaded");function s(){const n=i();return n==null?null:n.getClassification()}function i(){const n=t.getEntity(e.dataset);return n??null}return(n,l)=>r.unref(o)?(r.openBlock(),r.createElementBlock("div",je,[r.createElementVNode("ul",null,[(r.openBlock(!0),r.createElementBlock(r.Fragment,null,r.renderList(s(),(b,x)=>(r.openBlock(),r.createElementBlock("li",{key:x},[r.createVNode(ze,{"ifc-entity":i(),"classification-element":b},null,8,["ifc-entity","classification-element"])]))),128))])])):r.createCommentVNode("",!0)}}),Ue=p._export_sfc(Re,[["__scopeId","data-v-d2d82ae7"]]),Ye=c=>{if(H(c)){const t=c.object;if(t.fragment!=null&&c.face&&c.instanceId!=null){const e=t.fragment.getVertexBlockID(t.geometry,c.face.a),o=t.fragment.getItemID(c.instanceId,e).replace(/\..*/,"");return c.entity.highlight("selection",t,o),()=>c.entity.clearHighlight()}}return null},$e=c=>({name:c.filename,source:{url:c.file},type:"ifc",visible:!0}),Ge=(c,t)=>{var b,x,d,g,I,S,P,B,a;if(!H(c))return;const e=(b=c.features)==null?void 0:b.at(0);if(!e)return;t.has("IFC")||t.set("IFC",[]);const o=t.get("IFC"),{ifcProperties:s,itemProperties:i}=e,n="NULL",l=((x=i.Name)==null?void 0:x.value)??n;o.push({key:"Site",value:((g=(d=c.entity.object3d.userData)==null?void 0:d.dataset)==null?void 0:g.name)??n}),o.push({key:"IFCType",value:T.IfcCategoryMap[i.type]??n}),o.push({key:"Name",value:l}),o.push({key:"ID",value:i.expressID}),o.push({key:"GlobalId",value:((I=i.GlobalId)==null?void 0:I.value)??n}),((S=i.Description)==null?void 0:S.value)!=null&&o.push({key:"Description",value:i.Description.value}),((P=i.PredefinedType)==null?void 0:P.value)!=null&&o.push({key:"PredefinedType",value:i.PredefinedType.value}),((B=i.ObjectType)==null?void 0:B.value)!=null&&o.push({key:"ObjectType",value:i.ObjectType.value});for(const{name:f,parentName:h,value:u}of s)t.has(h)||t.set(h,[]),(a=t.get(h))==null||a.push({key:f,value:u})},qe=c=>{const t=c.dataset,e=t.config,o=p.getCoordinates(e.source.position??t.get("position")),s=new j({...e.source,at:o,name:t.name});return Promise.resolve(s)};class He{constructor(){y(this,"id","builtin-ifc-loader");y(this,"name","IFC")}async initialize(t){t.datasets.registerDatasetType("ifc",{attributeExtractor:Ge,entityBuilder:qe,fileExtensions:["ifc"],highlight:Ye,icon:"bi-building",loader:$e,name:"IFC",propertyView:Ue}),Q.registerInspector("IfcEntity",Me),await p.Fetcher.fetch("web-ifc.wasm").catch(e=>{console.warn("Could not load web-ifc.wasm",e)})}}const k=new p.Color;let We=class extends p.Loader{constructor(t){super(t),this.propertyNameMapping={},this.customPropertyMapping={}}load(t,e,o,s){const i=this,n=new p.FileLoader(this.manager);n.setPath(this.path),n.setResponseType("arraybuffer"),n.setRequestHeader(this.requestHeader),n.setWithCredentials(this.withCredentials),n.load(t,function(l){try{e(i.parse(l))}catch(b){s?s(b):console.error(b),i.manager.itemError(t)}},o,s)}setPropertyNameMapping(t){this.propertyNameMapping=t}setCustomPropertyNameMapping(t){this.customPropertyMapping=t}parse(t){function e(a,f=0){const h=/^ply([\s\S]*)end_header(\r\n|\r|\n)/;let u="";const m=h.exec(a);m!==null&&(u=m[1]);const _={comments:[],elements:[],headerLength:f,objInfo:""},v=u.split(/\r\n|\r|\n/);let w;function M(E,N){const C={type:E[0]};return C.type==="list"?(C.name=E[3],C.countType=E[1],C.itemType=E[2]):C.name=E[1],C.name in N&&(C.name=N[C.name]),C}for(let E=0;E<v.length;E++){let N=v[E];if(N=N.trim(),N==="")continue;const C=N.split(/\s+/),L=C.shift();switch(N=C.join(" "),L){case"format":_.format=C[0],_.version=C[1];break;case"comment":_.comments.push(N);break;case"element":w!==void 0&&_.elements.push(w),w={},w.name=C[0],w.count=parseInt(C[1]),w.properties=[];break;case"property":w.properties.push(M(C,B.propertyNameMapping));break;case"obj_info":_.objInfo=N;break;default:console.log("unhandled",L,C)}}return w!==void 0&&_.elements.push(w),_}function o(a,f){switch(f){case"char":case"uchar":case"short":case"ushort":case"int":case"uint":case"int8":case"uint8":case"int16":case"uint16":case"int32":case"uint32":return parseInt(a);case"float":case"double":case"float32":case"float64":return parseFloat(a)}}function s(a,f){const h={};for(let u=0;u<a.length;u++){if(f.empty())return null;if(a[u].type==="list"){const m=[],_=o(f.next(),a[u].countType);for(let v=0;v<_;v++){if(f.empty())return null;m.push(o(f.next(),a[u].itemType))}h[a[u].name]=m}else h[a[u].name]=o(f.next(),a[u].type)}return h}function i(){const a={indices:[],vertices:[],normals:[],uvs:[],faceVertexUvs:[],colors:[],faceVertexColors:[]};for(const f of Object.keys(B.customPropertyMapping))a[f]=[];return a}function n(a){const f=a.map(u=>u.name);function h(u){for(let m=0,_=u.length;m<_;m++){const v=u[m];if(f.includes(v))return v}return null}return{attrX:h(["x","px","posx"])||"x",attrY:h(["y","py","posy"])||"y",attrZ:h(["z","pz","posz"])||"z",attrNX:h(["nx","normalx"]),attrNY:h(["ny","normaly"]),attrNZ:h(["nz","normalz"]),attrS:h(["s","u","texture_u","tx"]),attrT:h(["t","v","texture_v","ty"]),attrR:h(["red","diffuse_red","r","diffuse_r"]),attrG:h(["green","diffuse_green","g","diffuse_g"]),attrB:h(["blue","diffuse_blue","b","diffuse_b"])}}function l(a,f){const h=i(),u=/end_header\s+(\S[\s\S]*\S|\S)\s*$/;let m,_;(_=u.exec(a))!==null?m=_[1].split(/\s+/):m=[];const v=new Xe(m);e:for(let w=0;w<f.elements.length;w++){const M=f.elements[w],E=n(M.properties);for(let N=0;N<M.count;N++){const C=s(M.properties,v);if(!C)break e;x(h,M.name,C,E)}}return b(h)}function b(a){let f=new p.BufferGeometry;a.indices.length>0&&f.setIndex(a.indices),f.setAttribute("position",new p.Float32BufferAttribute(a.vertices,3)),a.normals.length>0&&f.setAttribute("normal",new p.Float32BufferAttribute(a.normals,3)),a.uvs.length>0&&f.setAttribute("uv",new p.Float32BufferAttribute(a.uvs,2)),a.colors.length>0&&f.setAttribute("color",new p.Float32BufferAttribute(a.colors,3)),(a.faceVertexUvs.length>0||a.faceVertexColors.length>0)&&(f=f.toNonIndexed(),a.faceVertexUvs.length>0&&f.setAttribute("uv",new p.Float32BufferAttribute(a.faceVertexUvs,2)),a.faceVertexColors.length>0&&f.setAttribute("color",new p.Float32BufferAttribute(a.faceVertexColors,3)));for(const h of Object.keys(B.customPropertyMapping))a[h].length>0&&f.setAttribute(h,new p.Float32BufferAttribute(a[h],B.customPropertyMapping[h].length));return f.computeBoundingSphere(),f}function x(a,f,h,u){if(f==="vertex"){a.vertices.push(h[u.attrX],h[u.attrY],h[u.attrZ]),u.attrNX!==null&&u.attrNY!==null&&u.attrNZ!==null&&a.normals.push(h[u.attrNX],h[u.attrNY],h[u.attrNZ]),u.attrS!==null&&u.attrT!==null&&a.uvs.push(h[u.attrS],h[u.attrT]),u.attrR!==null&&u.attrG!==null&&u.attrB!==null&&(k.setRGB(h[u.attrR]/255,h[u.attrG]/255,h[u.attrB]/255,p.SRGBColorSpace),a.colors.push(k.r,k.g,k.b));for(const m of Object.keys(B.customPropertyMapping))for(const _ of B.customPropertyMapping[m])a[m].push(h[_])}else if(f==="face"){const m=h.vertex_indices||h.vertex_index,_=h.texcoord;m.length===3?(a.indices.push(m[0],m[1],m[2]),_&&_.length===6&&(a.faceVertexUvs.push(_[0],_[1]),a.faceVertexUvs.push(_[2],_[3]),a.faceVertexUvs.push(_[4],_[5]))):m.length===4&&(a.indices.push(m[0],m[1],m[3]),a.indices.push(m[1],m[2],m[3])),u.attrR!==null&&u.attrG!==null&&u.attrB!==null&&(k.setRGB(h[u.attrR]/255,h[u.attrG]/255,h[u.attrB]/255,p.SRGBColorSpace),a.faceVertexColors.push(k.r,k.g,k.b),a.faceVertexColors.push(k.r,k.g,k.b),a.faceVertexColors.push(k.r,k.g,k.b))}}function d(a,f){const h={};let u=0;for(let m=0;m<f.length;m++){const _=f[m],v=_.valueReader;if(_.type==="list"){const w=[],M=_.countReader.read(a+u);u+=_.countReader.size;for(let E=0;E<M;E++)w.push(v.read(a+u)),u+=v.size;h[_.name]=w}else h[_.name]=v.read(a+u),u+=v.size}return[h,u]}function g(a,f,h){function u(m,_,v){switch(_){case"int8":case"char":return{read:w=>m.getInt8(w),size:1};case"uint8":case"uchar":return{read:w=>m.getUint8(w),size:1};case"int16":case"short":return{read:w=>m.getInt16(w,v),size:2};case"uint16":case"ushort":return{read:w=>m.getUint16(w,v),size:2};case"int32":case"int":return{read:w=>m.getInt32(w,v),size:4};case"uint32":case"uint":return{read:w=>m.getUint32(w,v),size:4};case"float32":case"float":return{read:w=>m.getFloat32(w,v),size:4};case"float64":case"double":return{read:w=>m.getFloat64(w,v),size:8}}}for(let m=0,_=a.length;m<_;m++){const v=a[m];v.type==="list"?(v.countReader=u(f,v.countType,h),v.valueReader=u(f,v.itemType,h)):v.valueReader=u(f,v.type,h)}}function I(a,f){const h=i(),u=f.format==="binary_little_endian",m=new DataView(a,f.headerLength);let _,v=0;for(let w=0;w<f.elements.length;w++){const M=f.elements[w],E=M.properties,N=n(E);g(E,m,u);for(let C=0;C<M.count;C++){_=d(v,E),v+=_[1];const L=_[0];x(h,M.name,L,N)}}return b(h)}function S(a){let f=0,h=!0,u="";const m=[],_=new TextDecoder().decode(a.subarray(0,5)),v=/^ply\r\n/.test(_);do{const w=String.fromCharCode(a[f++]);w!==`
2
- `&&w!=="\r"?u+=w:(u==="end_header"&&(h=!1),u!==""&&(m.push(u),u=""))}while(h&&f<a.length);return v===!0&&f++,{headerText:m.join("\r")+"\r",headerLength:f}}let P;const B=this;if(t instanceof ArrayBuffer){const a=new Uint8Array(t),{headerText:f,headerLength:h}=S(a),u=e(f,h);if(u.format==="ascii"){const m=new TextDecoder().decode(a);P=l(m,u)}else P=I(t,u)}else P=l(t,e(t));return P}};class Xe{constructor(t){this.arr=t,this.i=0}empty(){return this.i>=this.arr.length}next(){return this.arr[this.i++]}}const V=class V extends p.Mesh{constructor(){super(...arguments);y(this,"isPickableFeatures",!0);y(this,"isPlyMesh",!0)}pickFeaturesFrom(e){if(this.geometry.hasAttribute("color")&&e.face){const o=this.geometry.getAttribute("color").array,s=e.face,n=[{color:new p.Color(o[s.a*3],o[s.a*3+1],o[s.a*3+2])}];return e.features=n,n}return[]}};y(V,"isPlyMesh",e=>p.isObject(e)&&e.isPlyMesh),y(V,"isPlyPickResult",e=>p.isObject(e)&&V.isPlyMesh(e==null?void 0:e.object));let O=V;class Ze extends R{constructor(e){super(new p.Group);y(this,"isPlyEntity",!0);y(this,"source");this.source=e}async preprocess(){const e=await p.Fetcher.fetchArrayBuffer(this.source.url),o=this.source.at.as(this.source.featureProjection).toVector3(),i=new We().parse(e),n=new p.MeshLambertMaterial({side:p.DoubleSide});i.hasAttribute("color")&&(n.vertexColors=!0),i.computeVertexNormals();const l=new O(i,n);l.name="plyModel",i.computeBoundingBox(),l.position.copy(o),l.updateWorldMatrix(!0,!0),this.object3d.add(l),this.onObjectCreated(l);const b=p.Fetcher.getContext(this.source.url);p.fillObject3DUserData(this,{filename:b.filename}),this.notifyChange(this.object3d)}}const Qe=c=>{const{dataset:t,instance:e}=c,o=t.config,s=p.getCoordinates(o.source.position??t.get("position")),i=new Ze({...o.source,at:s,featureProjection:e.referenceCrs});return Promise.resolve(i)},Je=(c,t)=>{var s;if(!O.isPlyPickResult(c))return;const e=(s=c.features)==null?void 0:s.at(0);if(!e)return;t.has("PLY")||t.set("PLY",[]),t.get("PLY").push({key:"Color",value:e.color})};class Ke{constructor(){y(this,"id","builtin-ply-loader");y(this,"name","PLY")}initialize(t){t.datasets.registerDatasetType("ply",{attributeExtractor:Je,entityBuilder:Qe,icon:"bi-file-earmark-binary",name:"PLY"})}}const et=c=>{const t=c.dataset.config,e=new K({source:new ee({url:`${t.source.url}/${t.source.filename}`})});return p.fillObject3DUserData(e,{filename:t.source.url}),Promise.resolve(e)};class tt{constructor(){y(this,"id","builtin-potree-loader");y(this,"name","Potree")}initialize(t){t.datasets.registerDatasetType("potree",{entityBuilder:et,icon:"fg-multipoint",name:"Potree Point Cloud"})}}class nt{constructor(){y(this,"id","builtin-tour");y(this,"name","Tour");y(this,"_camera",null);y(this,"_cameraCallback",null);y(this,"_context",null);y(this,"_tours",null)}initialize(t){this._context=t,t.events.addEventListener("ready",this.start.bind(this))}buildTours(){const t=this._camera,e=new F.Tour({tourName:"main",useModalOverlay:!0}),o=new F.Tour({tourName:"navigating",useModalOverlay:!0}),s=new F.Tour({tourName:"analyzing",useModalOverlay:!0}),i=[{action:()=>{var d;return(d=F.activeTour)==null?void 0:d.next()},text:"Next"},{action:()=>{var d;return(d=F.activeTour)==null?void 0:d.cancel()},secondary:!0,text:"Exit"}],n=()=>{var a,f;const d=(a=F.activeTour)==null?void 0:a.getCurrentStep(),g=d==null?void 0:d.getElement(),I=g==null?void 0:g.querySelector(".shepherd-text"),S=(f=F.activeTour)==null?void 0:f.steps;if(d==null||g==null||I==null||S==null)return;const P=document.createElement("div");P.className="progress mt-3",P.setAttribute("role","progressbar"),P.style.height="2px";const B=document.createElement("div");B.className="progress-bar bg-success",B.style.width=`${100*(S.indexOf(d)/S.length)}%`,P.appendChild(B),I.appendChild(P)},l=async(d,g)=>new Promise(I=>{const S=document.getElementById(d);if(S&&!S.classList.contains("active")&&S.click(),document.querySelector(g))return I(document.querySelector(g));const P=new MutationObserver(()=>{document.querySelector(g)&&(P.disconnect(),I(document.querySelector(g)))});P.observe(document.body,{childList:!0,subtree:!0})});e.addStep({buttons:[{action:()=>{var d;(d=F.activeTour)==null||d.complete(),o.show(0)},text:"Navigating"},{action:()=>{var d;(d=F.activeTour)==null||d.complete(),s.show(0)},text:"Analyzing data"},{action:()=>{var d;return(d=F.activeTour)==null?void 0:d.cancel()},secondary:!0,text:"Exit"}],cancelIcon:{enabled:!0,label:"Exit tutorial"},id:"example-step",text:"<p>Welcome to <strong>Piero</strong>, the Giro3D application.<br/>We can guide you through the different features.</p>",title:"Welcome!",when:{show:n}}),o.addStep({attachTo:{element:"#main-view",on:"bottom"},buttons:i,id:"view",text:"<p>This is the <b>main view</b>.</p><p>Giro3D natively supports a broad range of data sources, from 2D raster and vector data, to 3D point clouds and tilesets.</p><p>Piero adds support for CityJSON and IFC files.</p>",when:{show:n}}),o.addStep({attachTo:{element:"#main-view",on:"bottom"},buttons:i,id:"navigate",text:'<p>This application integrates <a href="https://github.com/yomotsu/camera-controls">camera-controls</a>, a camera control for three.js.</p><p><b>Click</b> to move the camera. <b>Right-click</b> to rotate around a point. <b>Scroll</b> to zoom in or out.</p>',when:{hide:()=>{this._cameraCallback&&t.removeEventListener("interaction-end",this._cameraCallback),this._cameraCallback=null},show:()=>{let d=0;this._cameraCallback=()=>{var g;d+=1,d>2&&((g=F.activeTour)==null||g.next())},t.addEventListener("interaction-end",this._cameraCallback),n()}}}),o.addStep({attachTo:{element:"#toolbar",on:"right"},beforeShowPromise:()=>l("toolbar-datasets","#datasets-drop-zone"),buttons:i,id:"toolbar-layers",text:"<p>Giro3D supports multiple datasets.</p><p>You can toggle datasets as you wish with the <b>Datasets</b> panel.</p>",when:{show:n}}),o.addStep({attachTo:{element:"#basemap-list",on:"right"},beforeShowPromise:()=>l("toolbar-datasets","#datasets-drop-zone"),buttons:i,id:"basemaps",text:"<p><b>Basemaps</b> are color and elevation layers that make the basic shape and aspect of the <b>Map</b>.</p>",when:{show:n}}),o.addStep({attachTo:{element:"#overlay-list",on:"right"},beforeShowPromise:()=>l("toolbar-datasets","#datasets-drop-zone"),buttons:i,id:"overlays",text:"<p><b>Overlays</b> are vector layers in various formats (WFS, GML, GeoJSON...).</p>",when:{show:n}}),o.addStep({attachTo:{element:"#dataset-list",on:"right"},beforeShowPromise:()=>l("toolbar-datasets","#datasets-drop-zone"),buttons:i,id:"layers",text:"<p>The <b>Datasets</b> panel contains all 3D objects in the scene.</><p>You can toggle their visibility and delete them.<p><p>Most objects leverage Giro3D's adaptive resolution to optimize their display.</p>",when:{show:n}}),o.addStep({attachTo:{element:"#datasets-drop-zone",on:"right"},beforeShowPromise:()=>l("toolbar-datasets","#datasets-drop-zone"),buttons:i,id:"adddata",text:"<p>You can add your own data from your computer by <b>dragging the file</b> into this page.</p><p>While you won't benefit from Giro3D's tiling mechanism, this can be a great way to quickly visualize datasets up to 100MB.</p><p>This application supports CityJSONs, IFCs, LAS/LAZs, CSV pointclouds, and simple GeoJSON features.</p>",when:{show:n}}),o.addStep({attachTo:{element:"#main-view",on:"bottom"},buttons:i,id:"attributes",text:"<p>By clicking on any feature on the map, you can see its <strong>Attribute table</strong>. Clickable features display a cursor when hovered.</p>",when:{show:n}}),o.addStep({attachTo:{element:"#search-place-autocomplete",on:"bottom"},buttons:[{action:()=>{var d;(d=F.activeTour)==null||d.complete(),s.show(0)},text:"Analyzing data"},{action:()=>{var d;return(d=F.activeTour)==null?void 0:d.complete()},secondary:!0,text:"Exit"}],id:"widgets",text:"Giro3D is highly extensible. Here we added a widget to search and navigate to locations based on the French address database.",when:{show:n}}),s.addStep({attachTo:{element:"#annotations-fieldset",on:"right"},beforeShowPromise:()=>l("toolbar-annotations","#annotations-fieldset"),buttons:i,id:"annotation",text:"<p>You can <strong>annotate</strong> any data displayed using Giro3D native tools.<br>Select the <strong>geometry</strong> of your annotation, and <strong>click</strong> on the scene to add points. <strong>Right-click</strong> to end the shape.</p>",when:{show:n}}),s.addStep({attachTo:{element:"#annotations-fieldset",on:"right"},beforeShowPromise:()=>l("toolbar-annotations","#annotations-fieldset"),buttons:i,id:"annotations",text:"You can download your annotations as GeoJSON files. You can also upload your own by dragging them into this panel.",when:{show:n}}),p.hasExperimentalFeature("measurements")&&s.addStep({attachTo:{element:"#panel-container",on:"right"},beforeShowPromise:()=>l("toolbar-measures","#measures-fieldset"),buttons:i,id:"measurements",text:"You can add <strong>measurements</strong> to easily get distances betwween objects.<br>Once started, moving the mouse will display the measure. <strong>Click</strong> to save the measurement. <strong>Right-click</strong> to end.",when:{show:n}}),s.addStep({attachTo:{element:"#panel-container",on:"right"},beforeShowPromise:()=>l("toolbar-analysis","#panel-container .card"),buttons:[{action:()=>{var d;return(d=F.activeTour)==null?void 0:d.complete()},text:"Done!"}],id:"analysis",text:"In the <strong>Analysis</strong> panel you'll find some advanced analysis tools.",when:{show:n}});const b=()=>{const d=new URL(document.URL);d.searchParams.delete("tourStep"),d.searchParams.set("tour","none"),window.history.replaceState({},"",d.toString())},x=d=>{const g=new URL(document.URL);let I="main";d.tour.id.startsWith("navigating")?I="navigating":d.tour.id.startsWith("analyzing")&&(I="analyzing"),g.searchParams.set("tour",I),g.searchParams.set("tourStep",d.step.id),window.history.replaceState({},"",g.toString())};return e.on("cancel",b),e.on("complete",b),e.on("show",x),o.on("cancel",b),o.on("complete",b),o.on("show",x),s.on("cancel",b),s.on("complete",b),s.on("show",x),{analyzingTour:s,mainTour:e,navigatingTour:o}}getTours(){return this._tours||(this._tours=this.buildTours()),this._tours}restart(){const{mainTour:t}=this.getTours();t.show(0)}start(){if(!this._context)throw new Error("module is not initialized");const{analyzingTour:t,mainTour:e,navigatingTour:o}=this.getTours();this._camera=this._context.view.getCameraController();const s=new URL(document.URL),i=s.searchParams.get("tour")??"main";if(i!=="none"){const n=s.searchParams.get("tourStep")??0;i==="navigating"?o.show(n):i==="analyzing"?t.show(n):e.show(n)}}}exports.ClippingBoxAnalysis=p.ClippingBoxAnalysis;exports.CrossSectionAnalysis=pe;exports.DownloadDataset=fe;exports.FloodingPlaneAnalysis=Ce;exports.IFCLoader=He;exports.PLYLoader=Ke;exports.PotreeLoader=tt;exports.Tour=nt;
1
+ "use strict";var Y=Object.defineProperty;var W=(r,t,e)=>t in r?Y(r,t,{enumerable:!0,configurable:!0,writable:!0,value:e}):r[t]=e;var d=(r,t,e)=>W(r,typeof t!="symbol"?t+"":t,e);Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const c=require("./Components.cjs.js"),i=require("vue"),T=require("@giro3d/giro3d/core/geographic/Coordinates"),H=require("pinia"),L=require("@giro3d/giro3d/entities/Entity3D"),Z=require("@giro3d/giro3d/utils/Fetcher"),J=require("@giro3d/giro3d/gui/EntityPanel"),_=require("openbim-components"),Q=require("@giro3d/giro3d/gui/EntityInspector"),z=require("ol"),X=require("ol/layer/Tile"),j=require("ol/proj"),K=require("ol/source"),ee=require("@giro3d/giro3d/entities/PointCloud"),te=require("@giro3d/giro3d/sources/PotreeSource"),w=require("shepherd.js"),ne={class:"input-group"},oe={class:"form-label"},se={class:"input-group"},ie=["value"],ae=["value"],re=["value"],le=i.defineComponent({__name:"CoordinateInput",props:{cursorManager:{},initialValue:{},instance:{},label:{},showZ:{type:Boolean}},emits:["update:coordinates"],setup(r,{emit:t}){var y,C,v;const e=r,n=i.ref(((y=e.initialValue)==null?void 0:y.x)??0),o=i.ref(((C=e.initialValue)==null?void 0:C.y)??0),s=i.ref(((v=e.initialValue)==null?void 0:v.z)??0),a=t,u=new c.Picker,h=i.ref(!1),m=()=>{a("update:coordinates",new T(c.nonNull(e.instance).referenceCrs,n.value,o.value,s.value))},l=()=>{const b=c.nonNull(e.instance);h.value=!0;const E=V=>{var I,S,k,A;const P=(S=(I=u.getFirstFeatureAt(b,V))==null?void 0:I.at(0))==null?void 0:S.point;P&&(n.value=P.x,o.value=P.y,s.value=P.z,(k=e.cursorManager)==null||k.setCursor("location"),(A=e.cursorManager)==null||A.setCursorLocation(P))},x=V=>{var I,S,k;(I=e.cursorManager)==null||I.setCursor(null),b.domElement.removeEventListener("click",x),b.domElement.removeEventListener("mousemove",E),h.value=!1;const P=(k=(S=u.getFirstFeatureAt(b,V))==null?void 0:S.at(0))==null?void 0:k.point;P&&(n.value=P.x,o.value=P.y,s.value=P.z,m())};b.domElement.addEventListener("mousemove",E),b.domElement.addEventListener("click",x)};function p(b){n.value=b,m()}function f(b){o.value=b,m()}function g(b){s.value=b,m()}return(b,E)=>(i.openBlock(),i.createElementBlock("div",ne,[i.createElementVNode("label",oe,i.toDisplayString(e.label),1),i.createElementVNode("div",se,[i.createElementVNode("input",{type:"number",class:"form-control",id:"plane-center-x",value:n.value.toFixed(2),onInput:E[0]||(E[0]=x=>p(Number.parseFloat(x.target.value)))},null,40,ie),i.createElementVNode("input",{type:"number",class:"form-control",id:"plane-center-y",value:o.value.toFixed(2),onInput:E[1]||(E[1]=x=>f(Number.parseFloat(x.target.value)))},null,40,ae),e.showZ===!0?(i.openBlock(),i.createElementBlock("input",{key:0,type:"number",class:"form-control",id:"plane-center-z",value:s.value.toFixed(2),onInput:E[2]||(E[2]=x=>g(Number.parseFloat(x.target.value)))},null,40,re)):i.createCommentVNode("",!0),e.instance!=null?(i.openBlock(),i.createBlock(c._sfc_main$1,{key:1,title:"Pick point in scene",icon:"fg-location",disabled:h.value,onClick:l,class:"btn-primary"},null,8,["disabled"])):i.createCommentVNode("",!0)])]))}}),$=H.defineStore("crossSection",()=>{const r=i.ref(0),t=i.shallowRef(),e=i.shallowRef(),n=i.ref(new c.Vector3(0,0,0)),o=i.ref(!1),s=i.ref(!1);function a(f){o.value=f}function u(f){r.value=f}function h(f){n.value=f}function m(f){s.value=f}function l(f){t.value=f}function p(f){e.value=f}return{center:n,cursorManager:e,enable:o,instance:t,orientation:r,setCenter:h,setCursorManager:p,setEnabled:a,setInstance:l,setOrientation:u,setShowHelper:m,showHelper:s}}),ce={class:"input-group"},ue={class:"input-group"},de={class:"input-group my-3"},pe=["value"],he={class:"input-group mb-3"},me=["value"],fe=i.defineComponent({__name:"CrossSection",setup(r){const t=$();function e(o){t.setOrientation(o)}const n=o=>{t.setCenter(o.toVector3())};return(o,s)=>(i.openBlock(),i.createElementBlock(i.Fragment,null,[i.createElementVNode("div",ce,[i.createVNode(c._sfc_main,{"model-value":i.unref(t).enable,"onUpdate:modelValue":s[0]||(s[0]=a=>i.unref(t).setEnabled(a)),id:"cross-section-enable",title:"Enable cross-section"},null,8,["model-value"]),s[4]||(s[4]=i.createElementVNode("label",{for:"cross-section-enable",class:"form-check-label"},"Enable cross section",-1))]),i.createElementVNode("div",ue,[i.createVNode(c._sfc_main,{"model-value":i.unref(t).showHelper,"onUpdate:modelValue":s[1]||(s[1]=a=>i.unref(t).setShowHelper(a)),id:"cross-section-helper-enable",title:"Show plane helper"},null,8,["model-value"]),s[5]||(s[5]=i.createElementVNode("label",{for:"cross-section-helper-enable",class:"form-check-label"},"Show plane helper",-1))]),i.createElementVNode("div",de,[s[6]||(s[6]=i.createElementVNode("label",{for:"plane-orientation-range",class:"form-check-label"},"Plane orientation (0-360°)",-1)),i.createElementVNode("input",{id:"plane-orientation-range",title:"Altitude",type:"range",class:"form-range",min:"0",step:"0.1",max:"360",value:i.unref(t).orientation,onInput:s[2]||(s[2]=a=>e(Number.parseFloat(a.target.value)))},null,40,pe)]),i.createElementVNode("div",he,[i.createElementVNode("input",{type:"number",class:"form-control",id:"plane-orientation-number",step:"0.1",value:i.unref(t).orientation,onInput:s[3]||(s[3]=a=>e(Number.parseFloat(a.target.value)))},null,40,me)]),s[7]||(s[7]=i.createElementVNode("hr",null,null,-1)),i.unref(t).instance!=null?(i.openBlock(),i.createBlock(le,{key:0,label:"Center","initial-value":i.unref(t).center,instance:i.unref(t).instance,"cursor-manager":i.unref(t).cursorManager,"onUpdate:coordinates":n},null,8,["initial-value","instance","cursor-manager"])):i.createCommentVNode("",!0)],64))}});class ge extends c.Object3D{constructor(){super();const t=new c.Mesh(new c.PlaneGeometry(50,100,1,10),new c.MeshBasicMaterial({color:"#6bf904",depthTest:!1,depthWrite:!1,wireframe:!0})),e=new c.ArrowHelper(new c.Vector3(0,1,0),new c.Vector3(0,0,0),200);t.renderOrder=999,t.rotateY(Math.PI/2),this.add(t),this.add(e),this.updateMatrixWorld(!0)}}class ye{constructor(t){d(this,"_clippingPlanes",null);d(this,"_helper");d(this,"_instance");d(this,"_store",$());this.context=t,t.events.addEventListener("ready",()=>{this._instance=t.view.getInstance();const e=t.configuration.default_crs,n=t.configuration.analysis.cross_section;this._store.setCursorManager(t.view.getSceneCursorManager()),this._store.setInstance(t.view.getInstance()),this._store.setOrientation(n.orientation);const o=n.pivot,s=new T(o.crs??e,o.x,o.y,0).as(e);this._store.setCenter(s.toVector3()),this._instance.addEventListener("entity-added",this.updateCrossSection.bind(this)),this._store.$onAction(({after:a,name:u})=>{a(()=>{switch(u){case"setCenter":case"setEnabled":case"setOrientation":case"setShowHelper":this.updateCrossSection(),this.updateHelper(),this.showHelper(this._store.showHelper);break}})}),this.updateCrossSection(),this.updateHelper(),this.showHelper(this._store.showHelper)})}dispose(){}createHelperIfNecessary(){this._store.showHelper&&this._helper==null&&this._instance&&(this._helper=new ge,this._instance.add(this._helper).catch(console.error))}showHelper(t){this.createHelperIfNecessary(),this._helper&&(this._helper.visible=t),this.updateHelper()}updateCrossSection(){if(this._store.enable){const t=[],e=c.MathUtils.DEG2RAD*this._store.orientation,n=Math.cos(e),o=Math.sin(e),s=new c.Vector3(n,o,0),a=new c.Plane(s,0).distanceToPoint(this._store.center),u=new c.Plane(s,-a);t.push(u),this._clippingPlanes=t}else this._clippingPlanes=null;this.updateEntities()}updateEntities(){const t=this.context.view.getInstance();for(const e of t.getEntities())L.isEntity3D(e)&&(e.clippingPlanes=this._clippingPlanes);t.notifyChange()}updateHelper(){var t,e,n,o;this.createHelperIfNecessary(),(t=this._helper)==null||t.position.copy(this._store.center),(e=this._helper)==null||e.updateMatrixWorld(!0),(n=this._helper)==null||n.setRotationFromAxisAngle(new c.Vector3(0,0,1),c.MathUtils.degToRad(this._store.orientation)),(o=this._instance)==null||o.notifyChange()}}class be{constructor(){d(this,"id","builtin-cross-section-analysis");d(this,"name","Cross section");d(this,"_manager",null)}initialize(t){t.analysis.registerTool({component:fe,icon:"bi-circle-half",name:"Cross section"}),this._manager=new ye(t)}}function D(r){if("url"in r.config&&typeof r.config.url=="string")return r.config.url;if("source"in r.config&&typeof r.config.source=="object"&&"url"in r.config.source&&typeof r.config.source.url=="string")return r.config.source.url}function _e(r){return D(r)!=null}class ve{constructor(){d(this,"id","builtin-download-dataset");d(this,"name","Download dataset");d(this,"_context",null)}initialize(t){this._context=t,t.datasets.registerDatasetAction({action:this.download.bind(this),icon:"bi-download",predicate:_e,title:"Download dataset"})}download(t){const e=D(t);if(e==null){console.warn("invalid");return}const n=this._context,o=new URL(e).pathname.split("/"),s=o[o.length-1];console.info(`download ${D(t)}`),c.Fetcher.fetch(e).then(a=>{if(!a.ok)throw new Error(`${a.status} ${a.statusText}`);return a.blob()}).then(a=>{c.Download.downloadBlob(a,s),n.notifications.pushNotification({level:"success",text:s,title:"Download successful"})}).catch(a=>{console.error(a);const u=a instanceof Error?a.message:"Download failed";n.notifications.pushNotification({level:"error",text:u,title:t.name})})}}const R=H.defineStore("floodingPlane",()=>{const r=i.ref(0),t=i.ref(!1);function e(){return r.value}function n(s){r.value=s}function o(s){t.value=s}return{enable:t,getHeight:e,setEnabled:o,setHeight:n}}),we={class:"input-group mb-3"},Ce={class:"input-group mb-3"},Ee=["value"],xe={class:"input-group mb-3"},Pe=["value"],Ie=i.defineComponent({__name:"FloodingPlane",setup(r){const t=R();function e(n){t.setHeight(n)}return(n,o)=>(i.openBlock(),i.createElementBlock(i.Fragment,null,[i.createElementVNode("div",we,[i.createVNode(c._sfc_main,{"model-value":i.unref(t).enable,"onUpdate:modelValue":o[0]||(o[0]=s=>i.unref(t).setEnabled(s)),id:"flooding-plane-enable",title:"foo"},null,8,["model-value"]),o[3]||(o[3]=i.createElementVNode("label",{for:"flooding-plane-enable",class:"form-label"},"Enable flooding plane",-1))]),i.createElementVNode("div",Ce,[o[5]||(o[5]=i.createElementVNode("label",{for:"flooding-altitude-range",class:"form-label"},"Altitude",-1)),i.createElementVNode("input",{id:"flooding-altitude-range",title:"Altitude",type:"range",class:"form-range",min:"0",step:"0.1",max:"500",value:i.unref(t).getHeight(),onInput:o[1]||(o[1]=s=>e(Number.parseFloat(s.target.value)))},null,40,Ee),i.createElementVNode("div",xe,[i.createElementVNode("input",{type:"number",class:"form-control",id:"flooding-altitude-number",step:"0.1",value:i.unref(t).getHeight(),onInput:o[2]||(o[2]=s=>e(Number.parseFloat(s.target.value)))},null,40,Pe),o[4]||(o[4]=i.createElementVNode("span",{class:"input-group-text"},"m",-1))])])],64))}});class Se{constructor(){d(this,"geometry");d(this,"material");d(this,"object3D");d(this,"_height");this.geometry=new c.PlaneGeometry(1,1,1,1),this.material=new c.MeshBasicMaterial({color:43690,opacity:.5,transparent:!0}),this.object3D=new c.Mesh(this.geometry,this.material),this.object3D.renderOrder=2,this.visible=!1,this._height=0}set height(t){this._height=t}get height(){return this._height}set visible(t){this.object3D.visible=t}get visible(){return this.object3D.visible}dispose(){this.object3D.removeFromParent(),this.geometry.dispose(),this.material.dispose()}setPosition(t,e,n,o,s){this.object3D.scale.set(o,s,1),this.object3D.position.set(t,e,n),this.object3D.updateMatrixWorld()}}class ke{constructor(t){d(this,"_context");d(this,"_plane");d(this,"_store",R());this._context=t,this._plane=null,t.events.addEventListener("ready",()=>{this._store.$onAction(({after:e,name:n})=>{e(()=>{switch(n){case"setEnabled":this.updatePlane();break;case"setHeight":this.updatePlane();break}})})})}dispose(){this._plane&&(this._context.view.getInstance().remove(this._plane.object3D),this._plane.dispose())}async updatePlane(){this._plane||(this._plane=new Se,await this._context.view.getInstance().add(this._plane.object3D));const t=this._context.view.getBoundingBox(),e=t.getCenter(new c.Vector3),n=t.getSize(new c.Vector3);this._plane.visible=this._store.enable,this._plane.setPosition(e.x,e.y,this._store.getHeight(),n.x,n.y),this._context.view.getInstance().notifyChange()}}class Fe{constructor(){d(this,"id","builtin-flooding-plane-analysis");d(this,"name","Flooding plane");d(this,"_manager",null)}initialize(t){t.analysis.registerTool({component:Ie,icon:"bi-layers-half",name:"Flooding plane"}),this._manager=new ke(t)}}const Be=/^(\d+\.\d+)\s*,\s*(\d+\.\d+)$/;class Ne{constructor(){d(this,"name","Go to coordinates")}search(t){const e=Be.exec(t);if(e){const n=Number.parseFloat(e[1]),o=Number.parseFloat(e[2]),a={coordinates:T.WGS84(n,o),label:`${n.toFixed(6)}, ${o.toFixed(6)}`,provider:this};return Promise.resolve([a])}else return Promise.resolve([])}}class Te{constructor(){d(this,"id","builtin-coordinates-search");d(this,"name","Coordinates search")}initialize(t){t.search.registerProvider(new Ne)}}const Me=/[a-z]{3}/;class Le{constructor(){d(this,"name","Base adresse nationale (BAN)")}async search(t){if(!Me.test(t))return[];try{const n=(await c.Fetcher.fetchJson(`https://api-adresse.data.gouv.fr/search/?q=${t}`)).features.map(o=>{const s=o.properties,a=o.geometry,[u,h]=a.coordinates;return{coordinates:new T("EPSG:4326",u,h,0),label:s.label,provider:this}});return await Ve(n.map(o=>o.coordinates)),n}catch(e){if(e instanceof Z.HttpError)return[];throw e}}}async function Ve(r){const t=new URL("https://data.geopf.fr/altimetrie/1.0/calcul/alti/rest/elevation.json");t.searchParams.append("lon",r.map(n=>n.longitude).join("|")),t.searchParams.append("lat",r.map(n=>n.latitude).join("|")),t.searchParams.append("zonly","true"),t.searchParams.append("resource","ign_rge_alti_wld"),t.searchParams.append("delimiter","|"),t.searchParams.append("indent","false");const e=await c.Fetcher.fetchJson(t.toString());for(let n=0;n<r.length;n++)r[n].setAltitude(e.elevations[n]);return e}class De{constructor(){d(this,"id","builtin-geocoding-ban");d(this,"name","Base adresse nationale")}initialize(t){t.search.registerProvider(new Le)}}const F=new c.Matrix4,Oe={bbox:new c.MeshBasicMaterial({color:"#FFFF00",depthTest:!0,opacity:.2,transparent:!0}),selection:new c.MeshBasicMaterial({color:"#FF0000",depthTest:!1,opacity:.6,transparent:!0})},Ae=4186316022,ze=781010003,je=2655215786,He=3242617779,$e=919958153,Re=1307041759,U=1451395588,q=1883228015,Ue=[U,q],qe=[Ae,ze,je,He,$e,Re],G=r=>c.isObject(r)&&r.isIFCPickResult,B=class B extends L{constructor(e){super(new c.Group);d(this,"isIfcEntity",!0);d(this,"isPickableFeatures",!0);d(this,"type","IfcEntity");d(this,"_classificationCache");d(this,"_components");d(this,"_fragmentBoundingBox");d(this,"_fragmentClassifier");d(this,"_fragmentManager");d(this,"_ifcSelection");d(this,"_indexMap");d(this,"_model");d(this,"_source");this._source=e,this._components=new _.Components,this._components.ui.enabled=!1,this._components.scene=new _.SimpleScene(this._components),this._components.renderer=new _.SimpleRenderer(this._components,document.createElement("div")),this._components.camera=new _.SimpleCamera(this._components),this._components.raycaster=new _.SimpleRaycaster(this._components),this._components.init(),this._ifcSelection={bbox:{},selection:{}},this._indexMap={},this._classificationCache=null,this._fragmentBoundingBox=null}clearHighlight(e="selection"){for(const n of Object.keys(this._ifcSelection[e])){const o=this._fragmentManager.list[n],s=o==null?void 0:o.fragments[e];s!=null&&s.mesh.removeFromParent()}this.notifyChange(this),this._ifcSelection[e]={}}getBoundingBoxById(e){this.clearHighlight("bbox"),this.highlightById(e,"bbox");const n=this._fragmentBoundingBox;if(n===null)throw new Error("Must call initClassification before getBoundingBoxById");const o=this._fragmentManager;n.reset();const s=this._ifcSelection.bbox;if(!Object.keys(s).length)return;for(const p of Object.keys(s)){const g=o.list[p].fragments.bbox;n.addMesh(g.mesh)}const a=n.get(),{y:u,z:h}=a.min,{y:m,z:l}=a.max;return a.min.y=-l,a.max.y=-h,a.min.z=u,a.max.z=m,a.translate(this._model.position),this.clearHighlight("bbox"),a}getClassification(){if(this._classificationCache===null)throw new Error("Must call initClassification before getClassification");return this._classificationCache}getProperties(e){const n=[],o=this._model.properties;if(!o)return[];for(const s of this._indexMap[e]){const a=o[s];if(a==null)continue;const{name:u}=_.IfcPropertiesUtils.getEntityName(o,s);if(u!==null){if(a.type===U){const h=_.IfcPropertiesUtils.getPsetProps(o,s);if(h!==null)for(const m of h){if(o[m]==null)continue;const p=this.getProperty(m);p!==null&&n.push({parentName:u,...p})}}else if(a.type===q){const h=_.IfcPropertiesUtils.getQsetQuantities(o,s);if(h!==null)for(const m of h){const{key:l}=_.IfcPropertiesUtils.getQuantityValue(o,m);if(l===null)continue;const p=this.getProperty(m);p!==null&&n.push({parentName:u,...p})}}}}return n}getProperty(e){const n=this._model.properties;if(n===void 0)return null;const{name:o}=_.IfcPropertiesUtils.getEntityName(n,e);if(o===null)return null;const{value:s}=_.IfcPropertiesUtils.getQuantityValue(n,e);return{name:o,value:s}}highlight(e,n,o){this._ifcSelection[e][n.uuid]=new Set;const s=parseInt(o,10);this._ifcSelection[e][n.uuid].add(o),this.addComposites(e,n,s),this.regenerate(e,n.uuid);const a=n.fragment.group;if(a){const u=a.data[s][0];for(let h=0;h<u.length;h++){const m=u[h],l=a.keyFragments[m],p=this._fragmentManager.list[l];l in this._ifcSelection[e]||(this._ifcSelection[e][l]=new Set),this._ifcSelection[e][l].add(o),this.addComposites(e,p.mesh,s),this.regenerate(e,l)}}this.notifyChange(this)}highlightById(e,n="selection"){for(const o of Object.keys(e)){o in this._ifcSelection[n]||(this._ifcSelection[n][o]=new Set);const s=this._fragmentManager.list[o],a=new Set;for(const u of e[o])this._ifcSelection[n][o].add(u),a.add(parseInt(u,10));for(const u of a)this.addComposites(n,s.mesh,u);this.regenerate(n,o)}this.notifyChange(this)}pick(e,n){return super.pick(e,n).map(o=>({...o,entity:this,features:o.features,isIFCPickResult:!0,object:o.object}))}pickFeaturesFrom(e){var o,s,a;const n=e.object;if(n.fragment!=null&&e.instanceId!=null&&e.face){const u=n.fragment.getVertexBlockID(n.geometry,e.face.a),h=(o=n.fragment.getItemID(e.instanceId,u))==null?void 0:o.replace(/\..*/,"");if(h&&((a=(s=n.fragment.group)==null?void 0:s.properties)==null?void 0:a[h])!=null){const l=n.fragment.group.properties[h],f=[{ifcProperties:this.getProperties(h),itemProperties:l}];return e.features=f,f}}return[]}async initClassification(){this._classificationCache=await this.regenerateClassification(["storeys","entities"]),this._classificationCache.length===0&&(this._classificationCache=await this.regenerateClassification(["entities"])),this._fragmentBoundingBox=await this._components.tools.get(_.FragmentBoundingBox)}async preprocess(){const e=await c.Fetcher.fetchArrayBuffer(this._source.url);this._fragmentManager=await this._components.tools.get(_.FragmentManager),this._fragmentClassifier=await this._components.tools.get(_.FragmentClassifier);const n=new _.FragmentIfcLoader(this._components);n.settings.webIfc.COORDINATE_TO_ORIGIN=!0,n.settings.webIfc.OPTIMIZE_PROFILES=!0;const o=new Uint8Array(e);this._model=await n.load(o,this._source.name),this._model.rotateX(Math.PI/2);const s=new c.Vector3;if(this._source.at)this._source.at.toVector3(s),this._model.position.copy(s);else{const u=this._model.coordinationMatrix.clone().invert();s.applyMatrix4(u),this._model.position.set(s.x,-s.z,s.y)}this._model.updateWorldMatrix(!0,!0),this._model.updateMatrix(),this._model.updateMatrixWorld(!0),this.initializeEntityIndexes(),this._fragmentClassifier.byStorey(this._model),this._fragmentClassifier.byEntity(this._model),await this.initClassification(),this.object3d.add(this._model),this.onObjectCreated(this._model);const a=c.Fetcher.getContext(this._source.url);c.fillObject3DUserData(this,{filename:a.filename}),this.notifyChange(this.object3d)}addComposites(e,n,o){this.addHighlightToFragment(e,n.fragment);const s=n.fragment.composites[o];if(s)for(let a=1;a<s;a++){const u=_.toCompositeID(o,a);this._ifcSelection[e][n.uuid].add(u)}}addHighlightToFragment(e,n){if(!(e in n.fragments)){const o=n.addFragment(e,[Oe[e]]);n.blocks.count>1&&(o.setInstance(0,{ids:Array.from(n.ids),transform:F}),o.blocks.setVisibility(!1)),this._model.add(o.mesh),o.mesh.renderOrder=30,o.mesh.frustumCulled=!1,o.mesh.name=e,o.mesh.updateMatrixWorld(!0)}}initializeEntityIndexes(){this._indexMap={};const e=this._model.properties;if(e!==void 0)for(const n of qe)_.IfcPropertiesUtils.getRelationMap(e,n,(o,s)=>{const a=e[o];Ue.includes(a.type)||this.setEntityIndex(o);for(const u of s)this.setEntityIndex(u).add(o)})}regenerate(e,n){this.updateFragmentFill(e,n)}async regenerateClassification(e,n={}){const o=this._fragmentClassifier.get(),s=[],a=e[0],u=o[a];if(a==null||u==null)return s;for(const h of Object.keys(u)){const m={...n,[a]:[h]},l=await this._fragmentClassifier.find(m);if(Object.keys(l).length>0){const g=a[0].toUpperCase()+a.slice(1),y=await this.regenerateClassification(e.slice(1),m);s.push({children:y,fragments:l,name:h,treeItemName:g})}}return s}setEntityIndex(e){return e in this._indexMap||(this._indexMap[e]=new Set),this._indexMap[e]}updateFragmentFill(e,n){const o=this._ifcSelection[e][n],s=this._fragmentManager.list[n];if(s==null)return;const a=s.fragments[e];if(a==null)return;const u=s.mesh.parent;if(u==null)return;if(u.add(a.mesh),a.blocks.count>1)s.getInstance(0,F),a.setInstance(0,{ids:Array.from(s.ids),transform:F}),a.blocks.setVisibility(!0,o,!0);else{let m=0;for(const l of o){a.mesh.count=m+1;const{instanceID:p}=s.getInstanceAndBlockID(l);s.getInstance(p,F),a.setInstance(m,{ids:[l],transform:F}),m++}}}};d(B,"isIFCEntity",e=>c.isObject(e)&&e.isIfcEntity),d(B,"isIFCPickResult",e=>c.isObject(e)&&B.isIFCEntity(e.entity));let O=B;class Ge extends Q{constructor(t,e,n){super(t,e,n,{visibility:!0})}}const Ye={class:"d-flex"},We=["title"],Ze=["id"],Je={class:"list-unstyled border-start"},Qe=i.defineComponent({__name:"IfcSubtree",props:{classificationElement:{},ifcEntity:{}},setup(r){const t=r,e=c.MathUtils.generateUUID(),n=`#${e}`,o=i.ref(!1),s=c.useCameraStore(),u=c.useModuleStore().getModule(c.moduleId);function h(){if(u==null){console.warn("Cannot clip IFC element, ClippingBoxAnalysis module is not present");return}const p=t.ifcEntity.getBoundingBoxById(t.classificationElement.fragments);p&&!p.isEmpty()&&u.setClippingBox(p)}function m(){o.value=!0,t.ifcEntity.clearHighlight(),t.ifcEntity.highlightById(t.classificationElement.fragments),setTimeout(()=>o.value=!1,2e3)}function l(){const p=t.ifcEntity.getBoundingBoxById(t.classificationElement.fragments);p&&!p.isEmpty()&&s.lookTopDownAt(p)}return(p,f)=>{const g=i.resolveComponent("IfcSubtree",!0);return i.openBlock(),i.createElementBlock("div",null,[i.createElementVNode("div",Ye,[i.createElementVNode("span",{class:i.normalizeClass(["border rounded px-1 py-0 fw-normal",o.value?"text-danger border-danger":"text-secondary border-secondary"]),title:p.classificationElement.treeItemName},i.toDisplayString(p.classificationElement.treeItemName),11,We),p.classificationElement.children.length>0?(i.openBlock(),i.createBlock(c.IconList,{key:0},{default:i.withCtx(()=>[i.createVNode(c.IconListButton,{title:"Expand group",icon:"bi-chevron-down","data-bs-toggle":"collapse","data-bs-target":n,"aria-controls":i.unref(e),"aria-expanded":"true"},null,8,["aria-controls"])]),_:1})):i.createCommentVNode("",!0),i.createVNode(c._sfc_main$3,{class:i.normalizeClass(["label",o.value?"text-danger-emphasis":"text-muted"]),text:p.classificationElement.name,title:`Zoom to ${p.classificationElement.name}`,onClick:l},null,8,["class","text","title"]),i.createVNode(c.IconList,{class:"ms-1"},{default:i.withCtx(()=>[i.createVNode(c.IconListButton,{title:"Highlight",icon:"bi-highlighter",onClick:m}),i.unref(u)!=null?(i.openBlock(),i.createBlock(c.IconListButton,{key:0,title:"Clip to",icon:"bi-bounding-box",onClick:h})):i.createCommentVNode("",!0)]),_:1})]),p.classificationElement.children.length>0?(i.openBlock(),i.createElementBlock("div",{key:0,id:i.unref(e),class:"collapse show"},[i.createElementVNode("ul",Je,[(i.openBlock(!0),i.createElementBlock(i.Fragment,null,i.renderList(p.classificationElement.children,(y,C)=>(i.openBlock(),i.createElementBlock("li",{key:C},[i.createVNode(g,{"ifc-entity":t.ifcEntity,"classification-element":y},null,8,["ifc-entity","classification-element"])]))),128))])],8,Ze)):i.createCommentVNode("",!0)])}}}),Xe=c._export_sfc(Qe,[["__scopeId","data-v-804ca598"]]),Ke={key:0},et=i.defineComponent({__name:"IfcPropertyView",props:{dataset:{}},setup(r){const t=c.useDatasetStore(),e=r,n=c.refAndWatch(e.dataset,"isPreloaded");function o(){const a=s();return a==null?null:a.getClassification()}function s(){const a=t.getEntity(e.dataset);return a??null}return(a,u)=>i.unref(n)?(i.openBlock(),i.createElementBlock("div",Ke,[i.createElementVNode("ul",null,[(i.openBlock(!0),i.createElementBlock(i.Fragment,null,i.renderList(o(),(h,m)=>(i.openBlock(),i.createElementBlock("li",{key:m},[i.createVNode(Xe,{"ifc-entity":s(),"classification-element":h},null,8,["ifc-entity","classification-element"])]))),128))])])):i.createCommentVNode("",!0)}}),tt=c._export_sfc(et,[["__scopeId","data-v-d2d82ae7"]]),nt=r=>{if(G(r)){const t=r.object;if(t.fragment!=null&&r.face&&r.instanceId!=null){const e=t.fragment.getVertexBlockID(t.geometry,r.face.a),n=t.fragment.getItemID(r.instanceId,e).replace(/\..*/,"");return r.entity.highlight("selection",t,n),()=>r.entity.clearHighlight()}}return null},ot=r=>({name:r.filename,source:{url:r.file},type:"ifc",visible:!0}),st=(r,t)=>{var h,m,l,p,f,g,y,C,v;if(!G(r))return;const e=(h=r.features)==null?void 0:h.at(0);if(!e)return;t.has("IFC")||t.set("IFC",[]);const n=t.get("IFC"),{ifcProperties:o,itemProperties:s}=e,a="NULL",u=((m=s.Name)==null?void 0:m.value)??a;n.push({key:"Site",value:((p=(l=r.entity.object3d.userData)==null?void 0:l.dataset)==null?void 0:p.name)??a}),n.push({key:"IFCType",value:_.IfcCategoryMap[s.type]??a}),n.push({key:"Name",value:u}),n.push({key:"ID",value:s.expressID}),n.push({key:"GlobalId",value:((f=s.GlobalId)==null?void 0:f.value)??a}),((g=s.Description)==null?void 0:g.value)!=null&&n.push({key:"Description",value:s.Description.value}),((y=s.PredefinedType)==null?void 0:y.value)!=null&&n.push({key:"PredefinedType",value:s.PredefinedType.value}),((C=s.ObjectType)==null?void 0:C.value)!=null&&n.push({key:"ObjectType",value:s.ObjectType.value});for(const{name:b,parentName:E,value:x}of o)t.has(E)||t.set(E,[]),(v=t.get(E))==null||v.push({key:b,value:x})},it=r=>{const t=r.dataset,e=t.config,n=c.getCoordinates(e.source.position??t.get("position")),o=new O({...e.source,at:n,name:t.name});return Promise.resolve(o)};class at{constructor(){d(this,"id","builtin-ifc-loader");d(this,"name","IFC")}async initialize(t){t.datasets.registerDatasetType("ifc",{attributeExtractor:st,entityBuilder:it,fileExtensions:["ifc"],highlight:nt,icon:"bi-building",loader:ot,name:"IFC",propertyView:tt}),J.registerInspector("IfcEntity",Ge),await c.Fetcher.fetch("web-ifc.wasm").catch(e=>{console.warn("Could not load web-ifc.wasm",e)})}}const rt=i.defineComponent({__name:"OpenLayersMinimapComponent",props:{context:{}},setup(r){const t=i.ref(),e=i.shallowRef(),n=r,o=[[3e4,12],[5e4,10],[1e5,8],[4e5,6],[8e5,4]];function s(a){for(const[u,h]of o)if(a<u)return h;return o[o.length-1][1]}return i.onMounted(()=>{e.value=new z.Map({controls:[],layers:[new X({source:new K.OSM})],target:t.value,view:new z.View({center:j.fromLonLat([4,44]),projection:"EPSG:3857",zoom:5})});let a=new c.Vector3;const u=()=>{var g;const m=n.context.view.getInstance(),p=n.context.view.getCameraController().getCameraPosition(),f=p.camera.z;if(!p.camera.equals(a)){a=p.camera.clone();const y=new T(m.referenceCrs,a.x,a.y).as("EPSG:4326"),C=s(f),v=(g=e.value)==null?void 0:g.getView();v==null||v.setCenter(j.fromLonLat([y.longitude,y.latitude])),v==null||v.setZoom(C)}};n.context.events.addEventListener("updated",u);let h=!1;t.value&&(t.value.onclick=()=>{var m,l,p,f;h=!h,h?((m=t.value)==null||m.classList.add("collapsed"),(l=t.value)==null||l.classList.remove("expanded")):((p=t.value)==null||p.classList.remove("collapsed"),(f=t.value)==null||f.classList.add("expanded"))})}),i.onUnmounted(()=>{var a;(a=e.value)==null||a.dispose(),e.value=void 0}),(a,u)=>(i.openBlock(),i.createElementBlock("div",{ref_key:"target",ref:t,title:"Toggle minimap",class:"minimap expanded"},null,512))}}),lt=c._export_sfc(rt,[["__scopeId","data-v-38ee3c04"]]);class ct{constructor(){d(this,"id","builtin-minimap-openlayers");d(this,"name","Minimap")}initialize(t){t.widgets.addWidget({component:lt,id:"minimap-ol"})}}const N=class N extends c.Mesh{constructor(){super(...arguments);d(this,"isPickableFeatures",!0);d(this,"isPlyMesh",!0)}pickFeaturesFrom(e){if(this.geometry.hasAttribute("color")&&e.face){const n=this.geometry.getAttribute("color").array,o=e.face,a=[{color:new c.Color(n[o.a*3],n[o.a*3+1],n[o.a*3+2])}];return e.features=a,a}return[]}};d(N,"isPlyMesh",e=>c.isObject(e)&&e.isPlyMesh),d(N,"isPlyPickResult",e=>c.isObject(e)&&N.isPlyMesh(e==null?void 0:e.object));let M=N;class ut extends L{constructor(e){super(new c.Group);d(this,"isPlyEntity",!0);d(this,"source");this.source=e}async preprocess(){const e=await c.Fetcher.fetchArrayBuffer(this.source.url),n=this.source.at.as(this.source.featureProjection).toVector3(),s=new c.PLYLoader().parse(e),a=new c.MeshLambertMaterial({side:c.DoubleSide});s.hasAttribute("color")&&(a.vertexColors=!0),s.computeVertexNormals();const u=new M(s,a);u.name="plyModel",s.computeBoundingBox(),u.position.copy(n),u.updateWorldMatrix(!0,!0),this.object3d.add(u),this.onObjectCreated(u);const h=c.Fetcher.getContext(this.source.url);c.fillObject3DUserData(this,{filename:h.filename}),this.notifyChange(this.object3d)}}const dt=r=>{const{dataset:t,instance:e}=r,n=t.config,o=c.getCoordinates(n.source.position??t.get("position")),s=new ut({...n.source,at:o,featureProjection:e.referenceCrs});return Promise.resolve(s)},pt=(r,t)=>{var o;if(!M.isPlyPickResult(r))return;const e=(o=r.features)==null?void 0:o.at(0);if(!e)return;t.has("PLY")||t.set("PLY",[]),t.get("PLY").push({key:"Color",value:e.color})};class ht{constructor(){d(this,"id","builtin-ply-loader");d(this,"name","PLY")}initialize(t){t.datasets.registerDatasetType("ply",{attributeExtractor:pt,entityBuilder:dt,icon:"bi-file-earmark-binary",name:"PLY"})}}const mt=r=>{r.addEventListener("object-created",t=>{t.obj.traverse(n=>{var o;((o=n.userData)==null?void 0:o.class)==="IfcSpace"&&(n.visible=!1)})})};class ft{constructor(){d(this,"id","builtin-post-process-entities");d(this,"name","Post-process 3D Tiles");d(this,"_alreadyProcessedEntities",new Set);d(this,"_processings",[mt])}initialize(t){t.events.addEventListener("ready",()=>{const e=t.view.getInstance();e.addEventListener("entity-added",()=>this.processEntities(e)),this.processEntities(e)})}processEntities(t){for(const e of t.getEntities())this._alreadyProcessedEntities.has(e.id)||(L.isEntity3D(e)&&this.processEntity(e,t),this._alreadyProcessedEntities.add(e.id))}processEntity(t,e){const n={instance:e};for(const o of this._processings)o(t,n)}}const gt=r=>{const t=r.dataset.config,e=new ee({source:new te({url:`${t.source.url}/${t.source.filename}`})});return c.fillObject3DUserData(e,{filename:t.source.url}),Promise.resolve(e)};class yt{constructor(){d(this,"id","builtin-potree-loader");d(this,"name","Potree")}initialize(t){t.datasets.registerDatasetType("potree",{entityBuilder:gt,icon:"fg-multipoint",name:"Potree Point Cloud"})}}class bt{constructor(){d(this,"id","builtin-tour");d(this,"name","Tour");d(this,"_camera",null);d(this,"_cameraCallback",null);d(this,"_context",null);d(this,"_tours",null)}initialize(t){this._context=t,t.events.addEventListener("ready",this.start.bind(this))}buildTours(){const t=this._camera,e=new w.Tour({tourName:"main",useModalOverlay:!0}),n=new w.Tour({tourName:"navigating",useModalOverlay:!0}),o=new w.Tour({tourName:"analyzing",useModalOverlay:!0}),s=[{action:()=>{var l;return(l=w.activeTour)==null?void 0:l.next()},text:"Next"},{action:()=>{var l;return(l=w.activeTour)==null?void 0:l.cancel()},secondary:!0,text:"Exit"}],a=()=>{var v,b;const l=(v=w.activeTour)==null?void 0:v.getCurrentStep(),p=l==null?void 0:l.getElement(),f=p==null?void 0:p.querySelector(".shepherd-text"),g=(b=w.activeTour)==null?void 0:b.steps;if(l==null||p==null||f==null||g==null)return;const y=document.createElement("div");y.className="progress mt-3",y.setAttribute("role","progressbar"),y.style.height="2px";const C=document.createElement("div");C.className="progress-bar bg-success",C.style.width=`${100*(g.indexOf(l)/g.length)}%`,y.appendChild(C),f.appendChild(y)},u=async(l,p)=>new Promise(f=>{const g=document.getElementById(l);if(g&&!g.classList.contains("active")&&g.click(),document.querySelector(p))return f(document.querySelector(p));const y=new MutationObserver(()=>{document.querySelector(p)&&(y.disconnect(),f(document.querySelector(p)))});y.observe(document.body,{childList:!0,subtree:!0})});e.addStep({buttons:[{action:()=>{var l;(l=w.activeTour)==null||l.complete(),n.show(0)},text:"Navigating"},{action:()=>{var l;(l=w.activeTour)==null||l.complete(),o.show(0)},text:"Analyzing data"},{action:()=>{var l;return(l=w.activeTour)==null?void 0:l.cancel()},secondary:!0,text:"Exit"}],cancelIcon:{enabled:!0,label:"Exit tutorial"},id:"example-step",text:"<p>Welcome to <strong>Piero</strong>, the Giro3D application.<br/>We can guide you through the different features.</p>",title:"Welcome!",when:{show:a}}),n.addStep({attachTo:{element:"#main-view",on:"bottom"},buttons:s,id:"view",text:"<p>This is the <b>main view</b>.</p><p>Giro3D natively supports a broad range of data sources, from 2D raster and vector data, to 3D point clouds and tilesets.</p><p>Piero adds support for CityJSON and IFC files.</p>",when:{show:a}}),n.addStep({attachTo:{element:"#main-view",on:"bottom"},buttons:s,id:"navigate",text:'<p>This application integrates <a href="https://github.com/yomotsu/camera-controls">camera-controls</a>, a camera control for three.js.</p><p><b>Click</b> to move the camera. <b>Right-click</b> to rotate around a point. <b>Scroll</b> to zoom in or out.</p>',when:{hide:()=>{this._cameraCallback&&t.removeEventListener("interaction-end",this._cameraCallback),this._cameraCallback=null},show:()=>{let l=0;this._cameraCallback=()=>{var p;l+=1,l>2&&((p=w.activeTour)==null||p.next())},t.addEventListener("interaction-end",this._cameraCallback),a()}}}),n.addStep({attachTo:{element:"#toolbar",on:"right"},beforeShowPromise:()=>u("toolbar-datasets","#datasets-drop-zone"),buttons:s,id:"toolbar-layers",text:"<p>Giro3D supports multiple datasets.</p><p>You can toggle datasets as you wish with the <b>Datasets</b> panel.</p>",when:{show:a}}),n.addStep({attachTo:{element:"#basemap-list",on:"right"},beforeShowPromise:()=>u("toolbar-datasets","#datasets-drop-zone"),buttons:s,id:"basemaps",text:"<p><b>Basemaps</b> are color and elevation layers that make the basic shape and aspect of the <b>Map</b>.</p>",when:{show:a}}),n.addStep({attachTo:{element:"#overlay-list",on:"right"},beforeShowPromise:()=>u("toolbar-datasets","#datasets-drop-zone"),buttons:s,id:"overlays",text:"<p><b>Overlays</b> are vector layers in various formats (WFS, GML, GeoJSON...).</p>",when:{show:a}}),n.addStep({attachTo:{element:"#dataset-list",on:"right"},beforeShowPromise:()=>u("toolbar-datasets","#datasets-drop-zone"),buttons:s,id:"layers",text:"<p>The <b>Datasets</b> panel contains all 3D objects in the scene.</><p>You can toggle their visibility and delete them.<p><p>Most objects leverage Giro3D's adaptive resolution to optimize their display.</p>",when:{show:a}}),n.addStep({attachTo:{element:"#datasets-drop-zone",on:"right"},beforeShowPromise:()=>u("toolbar-datasets","#datasets-drop-zone"),buttons:s,id:"adddata",text:"<p>You can add your own data from your computer by <b>dragging the file</b> into this page.</p><p>While you won't benefit from Giro3D's tiling mechanism, this can be a great way to quickly visualize datasets up to 100MB.</p><p>This application supports CityJSONs, IFCs, LAS/LAZs, CSV pointclouds, and simple GeoJSON features.</p>",when:{show:a}}),n.addStep({attachTo:{element:"#main-view",on:"bottom"},buttons:s,id:"attributes",text:"<p>By clicking on any feature on the map, you can see its <strong>Attribute table</strong>. Clickable features display a cursor when hovered.</p>",when:{show:a}}),n.addStep({attachTo:{element:"#search-place-autocomplete",on:"bottom"},buttons:[{action:()=>{var l;(l=w.activeTour)==null||l.complete(),o.show(0)},text:"Analyzing data"},{action:()=>{var l;return(l=w.activeTour)==null?void 0:l.complete()},secondary:!0,text:"Exit"}],id:"widgets",text:"Giro3D is highly extensible. Here we added a widget to search and navigate to locations based on the French address database.",when:{show:a}}),o.addStep({attachTo:{element:"#annotations-fieldset",on:"right"},beforeShowPromise:()=>u("toolbar-annotations","#annotations-fieldset"),buttons:s,id:"annotation",text:"<p>You can <strong>annotate</strong> any data displayed using Giro3D native tools.<br>Select the <strong>geometry</strong> of your annotation, and <strong>click</strong> on the scene to add points. <strong>Right-click</strong> to end the shape.</p>",when:{show:a}}),o.addStep({attachTo:{element:"#annotations-fieldset",on:"right"},beforeShowPromise:()=>u("toolbar-annotations","#annotations-fieldset"),buttons:s,id:"annotations",text:"You can download your annotations as GeoJSON files. You can also upload your own by dragging them into this panel.",when:{show:a}}),c.hasExperimentalFeature("measurements")&&o.addStep({attachTo:{element:"#panel-container",on:"right"},beforeShowPromise:()=>u("toolbar-measures","#measures-fieldset"),buttons:s,id:"measurements",text:"You can add <strong>measurements</strong> to easily get distances betwween objects.<br>Once started, moving the mouse will display the measure. <strong>Click</strong> to save the measurement. <strong>Right-click</strong> to end.",when:{show:a}}),o.addStep({attachTo:{element:"#panel-container",on:"right"},beforeShowPromise:()=>u("toolbar-analysis","#panel-container .card"),buttons:[{action:()=>{var l;return(l=w.activeTour)==null?void 0:l.complete()},text:"Done!"}],id:"analysis",text:"In the <strong>Analysis</strong> panel you'll find some advanced analysis tools.",when:{show:a}});const h=()=>{const l=new URL(document.URL);l.searchParams.delete("tourStep"),l.searchParams.set("tour","none"),window.history.replaceState({},"",l.toString())},m=l=>{const p=new URL(document.URL);let f="main";l.tour.id.startsWith("navigating")?f="navigating":l.tour.id.startsWith("analyzing")&&(f="analyzing"),p.searchParams.set("tour",f),p.searchParams.set("tourStep",l.step.id),window.history.replaceState({},"",p.toString())};return e.on("cancel",h),e.on("complete",h),e.on("show",m),n.on("cancel",h),n.on("complete",h),n.on("show",m),o.on("cancel",h),o.on("complete",h),o.on("show",m),{analyzingTour:o,mainTour:e,navigatingTour:n}}getTours(){return this._tours||(this._tours=this.buildTours()),this._tours}restart(){const{mainTour:t}=this.getTours();t.show(0)}start(){if(!this._context)throw new Error("module is not initialized");const{analyzingTour:t,mainTour:e,navigatingTour:n}=this.getTours();this._camera=this._context.view.getCameraController();const o=new URL(document.URL),s=o.searchParams.get("tour")??"main";if(s!=="none"){const a=o.searchParams.get("tourStep")??0;s==="navigating"?n.show(a):s==="analyzing"?t.show(a):e.show(a)}}}exports.ClippingBoxAnalysis=c.ClippingBoxAnalysis;exports.CoordinatesSearch=Te;exports.CrossSectionAnalysis=be;exports.DownloadDataset=ve;exports.FloodingPlaneAnalysis=Fe;exports.FrenchBanGeocoder=De;exports.IFCLoader=at;exports.OpenLayersMinimap=ct;exports.PLYLoader=ht;exports.PostProcessEntities=ft;exports.PotreeLoader=yt;exports.Tour=bt;
3
2
  //# sourceMappingURL=modules.cjs.js.map