@soonspacejs/plugin-measuring
Version:
Measuring plugin for SoonSpace.js
286 lines (285 loc) • 16.3 kB
JavaScript
import w from "soonspacejs";
import { LineBasicMaterial as x, DoubleSide as L, MeshBasicMaterial as C, SpriteMaterial as I, Sprite as B, BufferGeometry as m, Line as M, Mesh as D, Vector3 as p, QuadraticBezierCurve3 as H } from "three";
function F() {
const a = document.createElement("div");
return a.setAttribute("style", "height: 1in; visibility: hidden; position: absolute; margin: 0; padding: 0;"), document.body.appendChild(a), 0.0254 / a.clientHeight;
}
const E = {
m: 1,
mm: 1e-3,
cm: 0.01,
ft: 0.3048,
in: 0.0254,
pt: F()
}, P = (a) => a === 2 ? "²" : a === 3 ? "³" : "", y = (a, t = 1) => a + P(t), k = (a, t, e, s = 1) => (e == null && (e = t), e === t ? {
value: a,
unit: y(e)
} : {
value: a * Math.pow(E[t] / E[e], s),
unit: y(e) + P(s)
}), u = (a, t) => a.toFixed(t), N = "";
var T = /* @__PURE__ */ ((a) => (a.Distance = "Distance", a.Area = "Area", a.Angle = "Angle", a))(T || {});
const { utils: { randomString: b } } = w;
class d {
// save the last click time, in order to detect double click event
constructor(t) {
this.ssp = t;
const { viewport: { scene: e, camera: s } } = t;
this.scene = e, this.camera = s;
}
// lineWidth is ignored for Chrome on Windows, which is a known issue:
// https://github.com/mrdoob/three.js/issues/269
static LINE_MATERIAL = new x({ color: 16773120, linewidth: 2, opacity: 0.8, transparent: !0, side: L, depthWrite: !1, depthTest: !1 });
static MESH_MATERIAL = new C({ color: 8900346, transparent: !0, opacity: 0.8, side: L, depthWrite: !1, depthTest: !1 });
static MAX_DISTANCE = 500;
// when intersected object's distance is too far away, then ignore it
static OBJ_NAME = "object_for_measure" + b();
static LABEL_NAME = "label_for_measure" + b();
mode = null;
options = { unit: "m", precision: 2 };
scene;
camera;
spriteMaterial;
// 缓存之前的绘制对象
objectsStore = /* @__PURE__ */ new Set();
onCancel = () => {
};
onDone = () => {
};
pointMarkers = [];
// 边缘点标记
polyline;
// distance/area/angle 使用的绘制线段
faces;
// area
curve;
// angle 角度曲线
labels = [];
// poinode 标签
pointArray = [];
tempPointMarker;
// used to store temporary Points
tempLine;
// used to store temporary line, which is useful for drawing line/area/angle as mouse moves
tempLabel;
// used to store temporary label as mouse moves
lastClickTime;
get domElement() {
return this.ssp.domElement;
}
/**
* Starts the measurement
*/
start(t = "Distance", e = {}) {
const { unit: s = "m", precision: i = 2 } = e;
this.mode = t, this.options.unit = s, this.options.precision = i, this.ssp.signals.mouseMove.add(this.mousemove), this.ssp.signals.click.add(this.click), this.ssp.signals.dblClick.add(this.dblclick), this.ssp.signals.keyDown.add(this.keydown), this.pointArray = [], this.polyline = this.createLine(), this.scene.add(this.polyline), this.mode === "Area" && (this.faces = this.createFaces(), this.scene.add(this.faces)), this.domElement.style.cursor = "crosshair";
}
/**
* Ends the measurement
*/
close() {
this.mode !== null && (this.ssp.signals.mouseMove.remove(this.mousemove), this.ssp.signals.click.remove(this.click), this.ssp.signals.dblClick.remove(this.dblclick), this.ssp.signals.keyDown.remove(this.keydown), this.mode = null, this.domElement.style.cursor = "", this.pointArray = [], this.tempPointMarker && this.scene.remove(this.tempPointMarker), this.tempLine && this.scene.remove(this.tempLine), this.tempLabel && this.ssp.removeObject(this.tempLabel), this.pointMarkers.forEach((t) => this.scene.remove(t)), this.polyline && this.scene.remove(this.polyline), this.faces && this.scene.remove(this.faces), this.curve && this.scene.remove(this.curve), this.labels.forEach((t) => this.ssp.removeObject(t)), this.tempPointMarker = void 0, this.tempLine = void 0, this.tempLabel = void 0, this.pointMarkers = [], this.polyline = void 0, this.faces = void 0, this.curve = void 0, this.labels = [], this.ssp.render());
}
cancel() {
this.close(), this.onCancel();
}
clear() {
this.objectsStore.forEach((t) => this.ssp.removeObject(t)), this.objectsStore.clear();
}
/**
* Initializes point marker material
*/
initPointMarkerMaterial() {
const t = w.utils.textureLoader.load(N);
this.spriteMaterial = new I({
map: t,
depthTest: !1,
depthWrite: !1,
sizeAttenuation: !1,
opacity: 0.8
});
}
/**
* Creates point marker
*/
createPointMarker(t) {
this.spriteMaterial || this.initPointMarkerMaterial();
const e = 0.012, s = new B(this.spriteMaterial);
return s.scale.set(e, e, e), t && s.position.copy(t), s.name = d.OBJ_NAME, s;
}
/**
* Creates Line
*/
createLine(t = d.LINE_MATERIAL) {
const e = new m(), s = new M(e, t);
return s.frustumCulled = !1, s.name = d.OBJ_NAME, s;
}
/**
* Creates Mesh
*/
createFaces() {
const t = new m(), e = new D(t, d.MESH_MATERIAL);
return e.userData.vertices = [], e.frustumCulled = !1, e.name = d.OBJ_NAME, e;
}
/**
* Draw completed
*/
done() {
if (this.mode === null) return;
let t = !1;
const e = this.pointArray.length;
if (this.mode === "Area" && this.polyline)
if (e > 2) {
const s = this.pointArray[0];
this.polyline.geometry.dispose(), this.polyline.geometry = new m().setFromPoints([...this.pointArray, s]);
const i = this.calculateArea(this.pointArray), n = `${u(k(i, "m", this.options.unit, 2).value, this.options.precision)} ${this.getUnitString()}`, r = this.getBarycenter(this.pointArray), o = this.createLabel(n);
o.position.copy(r), o.element && (o.element.innerHTML = n), this.labels.push(o);
} else
t = !0;
if (this.mode === "Distance" && e < 2 && (t = !0), this.mode === "Angle" && this.polyline)
if (e >= 3) {
const s = this.pointArray[0], i = this.pointArray[1], n = this.pointArray[2], r = new p(s.x - i.x, s.y - i.y, s.z - i.z).normalize(), o = this.getAngleBisector(s, i, n), c = new p(n.x - i.x, n.y - i.y, n.z - i.z).normalize(), l = this.calculateAngle(s, i, n), v = `${u(l, this.options.precision)} ${this.getUnitString()}`, g = Math.min(s.distanceTo(i), n.distanceTo(i));
let h = g * 0.3, f = i.clone().add(new p(o.x * h, o.y * h, o.z * h));
const A = this.createLabel(v);
A.position.set(f.x, f.y, f.z), A.element.innerHTML = v, this.labels.push(A), h = g * 0.2, f = i.clone().add(new p(o.x * h, o.y * h, o.z * h));
const S = i.clone().add(new p(r.x * h, r.y * h, r.z * h)), z = i.clone().add(new p(c.x * h, c.y * h, c.z * h));
this.curve = this.createCurve(S, f, z), this.scene.add(this.curve);
} else
t = !0;
t || (this.pointMarkers.length > 0 && this.pointMarkers.forEach((s) => this.objectsStore.add(s)), this.polyline && this.objectsStore.add(this.polyline), this.faces && this.objectsStore.add(this.faces), this.curve && this.objectsStore.add(this.curve), this.labels.length > 0 && this.labels.forEach((s) => this.objectsStore.add(s)), this.pointMarkers = [], this.polyline = void 0, this.faces = void 0, this.curve = void 0, this.labels = []), this.close(), this.ssp.render(), this.onDone();
}
mousemove = (t) => {
if (this.mode === null) return;
const e = this.getClosestIntersection(t);
if (e) {
if (this.tempPointMarker ? this.tempPointMarker.position.set(e.x, e.y, e.z) : (this.tempPointMarker = this.createPointMarker(e), this.scene.add(this.tempPointMarker)), this.pointArray.length > 0) {
const s = this.pointArray[this.pointArray.length - 1], i = this.tempLine || this.createLine(), n = this.pointArray[0], r = this.pointArray[this.pointArray.length - 1];
if (this.mode === "Area" ? (i.geometry.dispose(), i.geometry = new m().setFromPoints([r, e, n])) : (i.geometry.dispose(), i.geometry = new m().setFromPoints([r, e])), this.mode === "Distance") {
const o = s.distanceTo(e), c = `${u(k(o, "m", this.options.unit).value, this.options.precision)} ${this.getUnitString()}`, l = new p((e.x + s.x) / 2, (e.y + s.y) / 2, (e.z + s.z) / 2);
this.addOrUpdateTempLabel(c, l);
}
this.tempLine || (this.scene.add(i), this.tempLine = i);
}
this.ssp.render();
}
};
dblclick = (t) => {
this.click(t), this.done();
};
click = (t) => {
if (this.mode === null) return;
const e = this.getClosestIntersection(t);
if (!e) return;
const s = Date.now();
if (this.lastClickTime && s - this.lastClickTime < 300)
return;
this.lastClickTime = s, this.pointArray.push(e);
const i = this.pointArray.length, n = this.createPointMarker(e);
if (this.pointMarkers.push(n), this.scene.add(n), this.polyline && (this.polyline.geometry.dispose(), this.polyline.geometry = new m().setFromPoints(this.pointArray), this.tempLabel && i > 1)) {
const r = this.pointArray[i - 2];
this.tempLabel.position.set((r.x + e.x) / 2, (r.y + e.y) / 2, (r.z + e.z) / 2), this.scene.add(this.tempLabel), this.labels.push(this.tempLabel), this.tempLabel = void 0;
}
if (this.mode === "Area" && this.faces) {
const r = this.faces.userData.vertices;
r.push(e), this.faces.geometry.dispose(), this.faces.geometry = new m().setFromPoints(r);
const o = r.length;
if (o > 2) {
const c = [];
for (let l = 1; l < o - 1; ++l)
c.push(0, l, l + 1);
this.faces.geometry.setIndex(c);
}
}
this.mode === "Angle" && this.pointArray.length >= 3 && this.done(), this.ssp.render();
};
keydown = (t) => {
t.key === "Enter" ? this.done() : t.key === "Escape" && this.cancel();
};
/**
* The the closest intersection
* @param e
*/
getClosestIntersection = (t) => {
if (this.mode === null) return;
let e = this.ssp.viewport.getIntersects(t);
return e && e.length > 0 && (e = e.filter((s) => s.object.name !== d.OBJ_NAME && s.object.visible), e.length > 0 && e[0].distance < d.MAX_DISTANCE) ? e[0].point : null;
};
/**
* Adds or update temp label and position
*/
addOrUpdateTempLabel(t, e) {
this.tempLabel || (this.tempLabel = this.createLabel(t)), this.tempLabel.position.set(e.x, e.y, e.z), this.tempLabel.element.innerHTML = t;
}
/**
* Creates label
*/
createLabel(t) {
const e = document.createElement("div");
e.innerHTML = t, e.style.padding = "3px 6px", e.style.color = "#fff", e.style.fontSize = "12px", e.style.position = "absolute", e.style.backgroundColor = "rgba(25, 25, 25, 0.3)", e.style.borderRadius = "12px", e.style.top = "0px", e.style.left = "0px";
const s = this.ssp.createPoiNode({ id: "label_id" + b(), element: e, type: "2D" });
return s.name = d.LABEL_NAME, s;
}
/**
* Creates the arc curve to indicate the angle in degree
*/
createCurve(t, e, s) {
const n = new H(t, e, s).getPoints(4), r = new m().setFromPoints(n), o = new M(r, d.LINE_MATERIAL);
return o.name = d.OBJ_NAME, o;
}
/**
* Calculates area
* TODO: for concave polygon, the value doesn't right, need to fix it
* @param points
*/
calculateArea(t) {
let e = 0;
for (let s = 0, i = 1, n = 2; n < t.length; i++, n++) {
const r = t[s].distanceTo(t[i]), o = t[i].distanceTo(t[n]), c = t[n].distanceTo(t[s]), l = (r + o + c) / 2;
e += Math.sqrt(l * (l - r) * (l - o) * (l - c));
}
return e;
}
/**
* Gets included angle of two lines in degree
*/
calculateAngle(t, e, s) {
const i = t, n = e, r = s, o = new p(i.x - n.x, i.y - n.y, i.z - n.z), c = new p(r.x - n.x, r.y - n.y, r.z - n.z);
return o.angleTo(c) * 180 / Math.PI;
}
/**
* Gets angle bisector of two lines
*/
getAngleBisector(t, e, s) {
const i = t, n = e, r = s, o = new p(i.x - n.x, i.y - n.y, i.z - n.z).normalize(), c = new p(r.x - n.x, r.y - n.y, r.z - n.z).normalize();
return new p(o.x + c.x, o.y + c.y, o.z + c.z).normalize();
}
/**
* Get the barycenter of points
*/
getBarycenter(t) {
const e = t.length;
let s = 0, i = 0, n = 0;
return t.forEach((r) => {
s += r.x, i += r.y, n += r.z;
}), new p(s / e, i / e, n / e);
}
/**
* Gets unit string for distance, area or angle
*/
getUnitString() {
return this.mode === "Distance" ? y(this.options.unit) : this.mode === "Area" ? `${y(this.options.unit, 2)}` : this.mode === "Angle" ? "°" : "";
}
/**
* Converts a number to a string with proper fraction digits
*/
numberToString(t) {
if (t < 1e-4)
return t.toString();
let e = 2;
return t < 0.01 ? e = 4 : t < 0.1 && (e = 3), t.toFixed(e);
}
}
export {
T as MeasuringMode,
d as default
};