build-dxf 0.1.6 → 0.1.8
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.
- package/package.json +2 -2
- package/src/build.js +955 -1395
- package/src/index.css +42 -0
- package/src/pages/DxfPreview.vue.d.ts +9 -0
- package/src/pages/ImagePreview.vue.d.ts +9 -0
- package/src/pages/Setting.vue.d.ts +6 -0
- package/src/utils/DxfSystem/components/Dxf.d.ts +1 -2
- package/src/utils/DxfSystem/utils/AxisAlignCorr02.d.ts +8 -0
- package/src/utils/DxfSystem/utils/index.d.ts +0 -1
- package/src/utils/LineSegment.d.ts +20 -5
- package/src/utils/modelScenario/SceneAutoGenerat.d.ts +7 -16
- package/src/utils/modelScenario/scenario.d.ts +1 -1
package/src/build.js
CHANGED
|
@@ -1,11 +1,11 @@
|
|
|
1
1
|
import * as THREE from "three";
|
|
2
|
-
import { EventDispatcher as EventDispatcher$1, Group as Group$1
|
|
2
|
+
import { EventDispatcher as EventDispatcher$1, Group as Group$1 } from "three";
|
|
3
3
|
import ClipperLib from "clipper-lib";
|
|
4
4
|
import Drawing from "dxf-writer";
|
|
5
5
|
import { OBJExporter } from "three/examples/jsm/exporters/OBJExporter.js";
|
|
6
|
+
import { Brush, Evaluator, SUBTRACTION } from "three-bvh-csg";
|
|
6
7
|
import { GLTFLoader } from "three/addons/loaders/GLTFLoader.js";
|
|
7
8
|
import { GLTFExporter } from "three/addons/exporters/GLTFExporter.js";
|
|
8
|
-
import { Brush, Evaluator, SUBTRACTION } from "three-bvh-csg";
|
|
9
9
|
const VITE_OSS_BASEURL = "https://image.cnyang.cn/common-assets/".endsWith("/") ? "https://image.cnyang.cn/common-assets/" : "https://image.cnyang.cn/common-assets//";
|
|
10
10
|
const DEFAULT_WALL_WIDTH = 0.12;
|
|
11
11
|
const DEFAULT_DOOR_HEIGHT = 2.1;
|
|
@@ -1309,6 +1309,38 @@ class UnionFindSet {
|
|
|
1309
1309
|
return sets;
|
|
1310
1310
|
}
|
|
1311
1311
|
}
|
|
1312
|
+
class LineIndexGenerator {
|
|
1313
|
+
index = 0;
|
|
1314
|
+
get size() {
|
|
1315
|
+
return this.index;
|
|
1316
|
+
}
|
|
1317
|
+
constructor(lines = []) {
|
|
1318
|
+
lines.forEach((key) => {
|
|
1319
|
+
const id = this.next();
|
|
1320
|
+
this.map.set(key, id);
|
|
1321
|
+
this.lineMap.set(id, key);
|
|
1322
|
+
});
|
|
1323
|
+
}
|
|
1324
|
+
next() {
|
|
1325
|
+
return this.index++;
|
|
1326
|
+
}
|
|
1327
|
+
reset() {
|
|
1328
|
+
this.index = 0;
|
|
1329
|
+
}
|
|
1330
|
+
map = /* @__PURE__ */ new Map();
|
|
1331
|
+
lineMap = /* @__PURE__ */ new Map();
|
|
1332
|
+
getIndex(key) {
|
|
1333
|
+
if (!this.map.has(key)) {
|
|
1334
|
+
const id = this.next();
|
|
1335
|
+
this.map.set(key, id);
|
|
1336
|
+
this.lineMap.set(id, key);
|
|
1337
|
+
}
|
|
1338
|
+
return this.map.get(key);
|
|
1339
|
+
}
|
|
1340
|
+
getLine(index2) {
|
|
1341
|
+
return this.lineMap.get(index2);
|
|
1342
|
+
}
|
|
1343
|
+
}
|
|
1312
1344
|
const set = /* @__PURE__ */ new Set();
|
|
1313
1345
|
function addLengthAsyncClear(line) {
|
|
1314
1346
|
if (set.size === 0) queueMicrotask(() => {
|
|
@@ -1485,7 +1517,7 @@ class LineSegment {
|
|
|
1485
1517
|
* @returns 线段的长度
|
|
1486
1518
|
*/
|
|
1487
1519
|
getLength() {
|
|
1488
|
-
return this.points[0].distance(this.points[1]);
|
|
1520
|
+
return this.points[0].distance(this.points[1], false);
|
|
1489
1521
|
}
|
|
1490
1522
|
/**
|
|
1491
1523
|
* 获取方向
|
|
@@ -1526,8 +1558,7 @@ class LineSegment {
|
|
|
1526
1558
|
const [q1, q2] = this.points;
|
|
1527
1559
|
const dir = new Point(q2.x - q1.x, q2.y - q1.y);
|
|
1528
1560
|
if (dir.x === 0 && dir.y === 0) {
|
|
1529
|
-
|
|
1530
|
-
return new LineSegment();
|
|
1561
|
+
throw new Error("投影目标线段的两个点不能重合");
|
|
1531
1562
|
}
|
|
1532
1563
|
const projectPoint = (point) => {
|
|
1533
1564
|
const pq = new Point(point.x - q1.x, point.y - q1.y);
|
|
@@ -1842,12 +1873,12 @@ class LineSegment {
|
|
|
1842
1873
|
/** 克隆
|
|
1843
1874
|
* @returns
|
|
1844
1875
|
*/
|
|
1845
|
-
clone() {
|
|
1876
|
+
clone(userData = true) {
|
|
1846
1877
|
const line = new LineSegment(
|
|
1847
1878
|
this.points[0].clone(),
|
|
1848
1879
|
this.points[1].clone()
|
|
1849
1880
|
);
|
|
1850
|
-
line.userData = cloneUserData(this.userData);
|
|
1881
|
+
if (userData) line.userData = cloneUserData(this.userData);
|
|
1851
1882
|
return line;
|
|
1852
1883
|
}
|
|
1853
1884
|
/**
|
|
@@ -1947,6 +1978,21 @@ class LineSegment {
|
|
|
1947
1978
|
}
|
|
1948
1979
|
return groups;
|
|
1949
1980
|
}
|
|
1981
|
+
/** 分组,通过点
|
|
1982
|
+
* @param lines
|
|
1983
|
+
* @returns
|
|
1984
|
+
*/
|
|
1985
|
+
static groupByPoint(lines) {
|
|
1986
|
+
const unionFindSet = new UnionFindSet(lines.length), map = /* @__PURE__ */ new Map();
|
|
1987
|
+
for (let i = 0; i < lines.length; i++) {
|
|
1988
|
+
lines[i].points.forEach((point) => {
|
|
1989
|
+
const id = point.hashCode();
|
|
1990
|
+
if (map.has(id)) unionFindSet.union(map.get(id), i);
|
|
1991
|
+
map.set(id, i);
|
|
1992
|
+
});
|
|
1993
|
+
}
|
|
1994
|
+
return unionFindSet.getAllSets().valueArray.map((list) => list.map((i) => lines[i]));
|
|
1995
|
+
}
|
|
1950
1996
|
/** 通过最大叶子路径,给线段分组,每条路径没有分叉
|
|
1951
1997
|
* @param lines
|
|
1952
1998
|
* @returns
|
|
@@ -1992,21 +2038,6 @@ class LineSegment {
|
|
|
1992
2038
|
}
|
|
1993
2039
|
return paths;
|
|
1994
2040
|
}
|
|
1995
|
-
/** 分组,通过点
|
|
1996
|
-
* @param lines
|
|
1997
|
-
* @returns
|
|
1998
|
-
*/
|
|
1999
|
-
static groupByPoint(lines) {
|
|
2000
|
-
const unionFindSet = new UnionFindSet(lines.length), map = /* @__PURE__ */ new Map();
|
|
2001
|
-
for (let i = 0; i < lines.length; i++) {
|
|
2002
|
-
lines[i].points.forEach((point) => {
|
|
2003
|
-
const id = point.hashCode();
|
|
2004
|
-
if (map.has(id)) unionFindSet.union(map.get(id), i);
|
|
2005
|
-
map.set(id, i);
|
|
2006
|
-
});
|
|
2007
|
-
}
|
|
2008
|
-
return unionFindSet.getAllSets().valueArray.map((list) => list.map((i) => lines[i]));
|
|
2009
|
-
}
|
|
2010
2041
|
/** 分组,通过方向
|
|
2011
2042
|
* @param lines
|
|
2012
2043
|
*/
|
|
@@ -2019,6 +2050,92 @@ class LineSegment {
|
|
|
2019
2050
|
}
|
|
2020
2051
|
return [parallelLines, verticalLines];
|
|
2021
2052
|
}
|
|
2053
|
+
/** 通过轴,对线段进行分组
|
|
2054
|
+
* @param lines
|
|
2055
|
+
* @param axisLine
|
|
2056
|
+
* @param errAngle
|
|
2057
|
+
* @returns
|
|
2058
|
+
*/
|
|
2059
|
+
static groupByAxis(lines, axisLine, errAngle = 1e-9) {
|
|
2060
|
+
const parallelLines = [], otherLines = [];
|
|
2061
|
+
for (let i = 0; i < lines.length; i++) {
|
|
2062
|
+
const line = lines[i];
|
|
2063
|
+
if (axisLine.isParallel(line, errAngle)) parallelLines.push(line);
|
|
2064
|
+
else otherLines.push(line);
|
|
2065
|
+
}
|
|
2066
|
+
return [parallelLines, otherLines];
|
|
2067
|
+
}
|
|
2068
|
+
/** 交叉轴分组
|
|
2069
|
+
* @description 通过交叉轴分组, 前提是使用groupByAxis对线段进行过轴分组,确保所有线段近乎平行
|
|
2070
|
+
* @param lines 需要分组的线段集合
|
|
2071
|
+
* @param crossAxis 交叉轴线段
|
|
2072
|
+
* @param distanceThreshold 交叉轴阈值
|
|
2073
|
+
* @returns
|
|
2074
|
+
*/
|
|
2075
|
+
static groupByCrossAxis(lines, crossAxis, distanceThreshold) {
|
|
2076
|
+
const center = crossAxis.center, normal = crossAxis.normal(), principalAxis = new LineSegment(
|
|
2077
|
+
center.clone().add(normal.clone().multiplyScalar(-1e3)),
|
|
2078
|
+
center.clone().add(normal.clone().multiplyScalar(1e3))
|
|
2079
|
+
), crossAxisMap = /* @__PURE__ */ new Map(), crossValueList = [], principalAxisStartMap = /* @__PURE__ */ new Map(), principalAxisEndMap = /* @__PURE__ */ new Map();
|
|
2080
|
+
lines.forEach((line) => {
|
|
2081
|
+
const value = crossAxis.projectValue(line.start);
|
|
2082
|
+
const point = crossAxis.projectPoint(line.start, false);
|
|
2083
|
+
const opt = { value, line, point, index: -1 };
|
|
2084
|
+
crossAxisMap.set(line, opt);
|
|
2085
|
+
crossValueList.push(opt);
|
|
2086
|
+
line.points.forEach((p, i) => {
|
|
2087
|
+
const value2 = principalAxis.projectValue(p);
|
|
2088
|
+
if (i === 0) principalAxisStartMap.set(line, value2);
|
|
2089
|
+
else principalAxisEndMap.set(line, value2);
|
|
2090
|
+
});
|
|
2091
|
+
});
|
|
2092
|
+
crossValueList.sort((a, b) => b.value - a.value).forEach((item, i) => item.index = i);
|
|
2093
|
+
const unionFindSet = new UnionFindSet(lines.length), lineIndexGenerator = new LineIndexGenerator(lines);
|
|
2094
|
+
function approach(preCrossOpt, crossOpt) {
|
|
2095
|
+
return preCrossOpt.point.distance(crossOpt.point) <= distanceThreshold;
|
|
2096
|
+
}
|
|
2097
|
+
function coincide(preCrossOpt, minPrinValue, maxPrinValue) {
|
|
2098
|
+
const prePrinStartV = principalAxisStartMap.get(preCrossOpt.line), prePrinEndV = principalAxisEndMap.get(preCrossOpt.line), minPrePrinValue = prePrinStartV < prePrinEndV ? prePrinStartV : prePrinEndV, maxPrePrinValue = prePrinStartV > prePrinEndV ? prePrinStartV : prePrinEndV;
|
|
2099
|
+
return minPrePrinValue >= minPrinValue && minPrePrinValue <= maxPrinValue || maxPrePrinValue >= minPrinValue && maxPrePrinValue <= maxPrinValue || minPrinValue >= minPrePrinValue && minPrinValue <= maxPrePrinValue || maxPrinValue >= minPrePrinValue && maxPrinValue <= maxPrePrinValue;
|
|
2100
|
+
}
|
|
2101
|
+
for (let i = 0; i < crossValueList.length; i++) {
|
|
2102
|
+
const crossOpt = crossValueList[i], { line } = crossOpt, prinStartV = principalAxisStartMap.get(line), prinEndV = principalAxisEndMap.get(line), minPrinValue = prinStartV < prinEndV ? prinStartV : prinEndV, maxPrinValue = prinStartV > prinEndV ? prinStartV : prinEndV;
|
|
2103
|
+
for (let j = i - 1; j >= 0; j--) {
|
|
2104
|
+
const pre = crossValueList[j];
|
|
2105
|
+
if (approach(pre, crossOpt)) {
|
|
2106
|
+
if (coincide(pre, minPrinValue, maxPrinValue)) {
|
|
2107
|
+
unionFindSet.union(lineIndexGenerator.getIndex(line), lineIndexGenerator.getIndex(pre.line));
|
|
2108
|
+
break;
|
|
2109
|
+
}
|
|
2110
|
+
} else break;
|
|
2111
|
+
}
|
|
2112
|
+
}
|
|
2113
|
+
let groups = unionFindSet.getAllSets().valueArray.map((group2) => group2.map((index2) => lineIndexGenerator.getLine(index2)));
|
|
2114
|
+
groups = groups.map((group2) => {
|
|
2115
|
+
if (group2.length <= 2) return [group2];
|
|
2116
|
+
const list = group2.map((line) => crossAxisMap.get(line)).sort((a, b) => a.value - b.value), start = list[0], end = list[list.length - 1];
|
|
2117
|
+
if (start.point.distance(end.point) < distanceThreshold) return [group2];
|
|
2118
|
+
const maxList = [...list].sort((a, b) => b.line.length() - a.line.length()), first = maxList[0], second = maxList[1], firstLength = first.line.length(), secondLength = second.line.length(), projLine = first.line.projectLineSegment(second.line), projLineLength = projLine.length();
|
|
2119
|
+
if (first.point.distance(second.point) > distanceThreshold || secondLength / firstLength > 0.5 && projLineLength / secondLength < 0.5) {
|
|
2120
|
+
const group1 = [first.line], group22 = [second.line];
|
|
2121
|
+
for (let i = 0; i < list.length; i++) {
|
|
2122
|
+
const opt = list[i];
|
|
2123
|
+
if (opt === first || opt === second) continue;
|
|
2124
|
+
const currentProjectLine = second.line.projectLineSegment(opt.line), len = currentProjectLine.length();
|
|
2125
|
+
if (second.point.distance(opt.point) <= distanceThreshold && len / secondLength > 0) group22.push(opt.line);
|
|
2126
|
+
else group1.push(opt.line);
|
|
2127
|
+
}
|
|
2128
|
+
return [group1, group22];
|
|
2129
|
+
}
|
|
2130
|
+
let index2 = 1;
|
|
2131
|
+
for (; index2 < list.length; index2++) {
|
|
2132
|
+
const { point } = list[index2];
|
|
2133
|
+
if (list[0].point.distance(point) > distanceThreshold) break;
|
|
2134
|
+
}
|
|
2135
|
+
return [list.slice(0, index2).map((o) => o.line), list.slice(index2).map((o) => o.line)];
|
|
2136
|
+
}).flat(1);
|
|
2137
|
+
return groups;
|
|
2138
|
+
}
|
|
2022
2139
|
/** 合并线段到最长线段
|
|
2023
2140
|
* @param lines
|
|
2024
2141
|
* @returns
|
|
@@ -3119,38 +3236,6 @@ class DoubleWallHelper {
|
|
|
3119
3236
|
return lines;
|
|
3120
3237
|
}
|
|
3121
3238
|
}
|
|
3122
|
-
class LineIndexGenerator {
|
|
3123
|
-
index = 0;
|
|
3124
|
-
get size() {
|
|
3125
|
-
return this.index;
|
|
3126
|
-
}
|
|
3127
|
-
constructor(lines = []) {
|
|
3128
|
-
lines.forEach((key) => {
|
|
3129
|
-
const id = this.next();
|
|
3130
|
-
this.map.set(key, id);
|
|
3131
|
-
this.lineMap.set(id, key);
|
|
3132
|
-
});
|
|
3133
|
-
}
|
|
3134
|
-
next() {
|
|
3135
|
-
return this.index++;
|
|
3136
|
-
}
|
|
3137
|
-
reset() {
|
|
3138
|
-
this.index = 0;
|
|
3139
|
-
}
|
|
3140
|
-
map = /* @__PURE__ */ new Map();
|
|
3141
|
-
lineMap = /* @__PURE__ */ new Map();
|
|
3142
|
-
getIndex(key) {
|
|
3143
|
-
if (!this.map.has(key)) {
|
|
3144
|
-
const id = this.next();
|
|
3145
|
-
this.map.set(key, id);
|
|
3146
|
-
this.lineMap.set(id, key);
|
|
3147
|
-
}
|
|
3148
|
-
return this.map.get(key);
|
|
3149
|
-
}
|
|
3150
|
-
getLine(index2) {
|
|
3151
|
-
return this.lineMap.get(index2);
|
|
3152
|
-
}
|
|
3153
|
-
}
|
|
3154
3239
|
function createPointVirtualGrid(lines) {
|
|
3155
3240
|
const grid = new PointVirtualGrid();
|
|
3156
3241
|
for (const seg of lines) {
|
|
@@ -5928,8 +6013,7 @@ class Dxf extends Component {
|
|
|
5928
6013
|
this.scale = scale;
|
|
5929
6014
|
this.shortLine = width * 0.4;
|
|
5930
6015
|
}
|
|
5931
|
-
/**
|
|
5932
|
-
* 预处理数据
|
|
6016
|
+
/** 预处理数据
|
|
5933
6017
|
* @param data
|
|
5934
6018
|
*/
|
|
5935
6019
|
preprocessing(data) {
|
|
@@ -7754,437 +7838,91 @@ function drawText$1(text, point, style, offset = 1e-3) {
|
|
|
7754
7838
|
}
|
|
7755
7839
|
}
|
|
7756
7840
|
}
|
|
7757
|
-
|
|
7758
|
-
|
|
7759
|
-
|
|
7760
|
-
|
|
7761
|
-
|
|
7762
|
-
|
|
7763
|
-
|
|
7764
|
-
|
|
7765
|
-
|
|
7766
|
-
|
|
7767
|
-
|
|
7768
|
-
|
|
7769
|
-
})
|
|
7770
|
-
|
|
7771
|
-
|
|
7772
|
-
|
|
7773
|
-
|
|
7774
|
-
|
|
7841
|
+
if (typeof globalThis !== "undefined" && typeof ProgressEvent === "undefined") {
|
|
7842
|
+
globalThis.ProgressEvent = class ProgressEvent extends EventTarget {
|
|
7843
|
+
type;
|
|
7844
|
+
lengthComputable;
|
|
7845
|
+
loaded;
|
|
7846
|
+
total;
|
|
7847
|
+
bubbles;
|
|
7848
|
+
cancelable;
|
|
7849
|
+
defaultPrevented;
|
|
7850
|
+
eventPhase;
|
|
7851
|
+
timeStamp;
|
|
7852
|
+
isTrusted;
|
|
7853
|
+
constructor(type, options = {}) {
|
|
7854
|
+
super();
|
|
7855
|
+
this.type = type;
|
|
7856
|
+
this.lengthComputable = options.lengthComputable || false;
|
|
7857
|
+
this.loaded = options.loaded || 0;
|
|
7858
|
+
this.total = options.total || 0;
|
|
7859
|
+
this.bubbles = false;
|
|
7860
|
+
this.cancelable = false;
|
|
7861
|
+
this.defaultPrevented = false;
|
|
7862
|
+
this.eventPhase = 0;
|
|
7863
|
+
this.timeStamp = Date.now();
|
|
7864
|
+
this.isTrusted = true;
|
|
7865
|
+
}
|
|
7866
|
+
preventDefault() {
|
|
7867
|
+
}
|
|
7868
|
+
stopPropagation() {
|
|
7869
|
+
}
|
|
7870
|
+
stopImmediatePropagation() {
|
|
7871
|
+
}
|
|
7872
|
+
};
|
|
7873
|
+
}
|
|
7874
|
+
let gltfLoader = GLTFExporter;
|
|
7875
|
+
let gltfExporter = GLTFLoader;
|
|
7876
|
+
let promise;
|
|
7877
|
+
function GltfInit() {
|
|
7878
|
+
if (promise) return promise;
|
|
7879
|
+
promise = new Promise(async (resolve) => {
|
|
7880
|
+
try {
|
|
7881
|
+
const { GLTFLoader: GLTFLoader2 } = await (typeof window !== "undefined" ? import("three/addons/loaders/GLTFLoader.js") : include("node-three-gltf", false));
|
|
7882
|
+
gltfLoader = GLTFLoader2;
|
|
7883
|
+
const { GLTFExporter: GLTFExporter2 } = await (typeof window !== "undefined" ? import("three/addons/exporters/GLTFExporter.js") : include("node-three-gltf", false));
|
|
7884
|
+
gltfExporter = GLTFExporter2;
|
|
7885
|
+
} catch (error) {
|
|
7886
|
+
console.error(error);
|
|
7887
|
+
}
|
|
7888
|
+
resolve(true);
|
|
7889
|
+
});
|
|
7890
|
+
return promise;
|
|
7891
|
+
}
|
|
7892
|
+
const gltf = {
|
|
7893
|
+
get GLTFLoader() {
|
|
7894
|
+
return gltfLoader;
|
|
7895
|
+
},
|
|
7896
|
+
get GLTFExporter() {
|
|
7897
|
+
return gltfExporter;
|
|
7775
7898
|
}
|
|
7776
|
-
|
|
7777
|
-
|
|
7778
|
-
|
|
7899
|
+
};
|
|
7900
|
+
let loader;
|
|
7901
|
+
class LoadModel {
|
|
7902
|
+
static _cache = /* @__PURE__ */ new Map();
|
|
7903
|
+
static _nameMap = /* @__PURE__ */ new Map();
|
|
7904
|
+
static addNameMap(mapName, name) {
|
|
7905
|
+
this._nameMap.set(mapName, name);
|
|
7779
7906
|
}
|
|
7780
|
-
|
|
7781
|
-
|
|
7782
|
-
if (this.
|
|
7783
|
-
this.
|
|
7784
|
-
|
|
7785
|
-
|
|
7786
|
-
|
|
7787
|
-
|
|
7788
|
-
|
|
7789
|
-
|
|
7907
|
+
// 获取模型
|
|
7908
|
+
static loadGlb(name) {
|
|
7909
|
+
if (this._nameMap.has(name)) name = this._nameMap.get(name);
|
|
7910
|
+
if (this._cache.has(name)) return this._cache.get(name);
|
|
7911
|
+
if (!loader) loader = new gltf.GLTFLoader();
|
|
7912
|
+
const promise2 = new Promise(async (resolve) => {
|
|
7913
|
+
const response = await fetch(`${VITE_OSS_BASEURL}models/${name}.glb`);
|
|
7914
|
+
if (!response.ok) {
|
|
7915
|
+
resolve(null);
|
|
7916
|
+
return;
|
|
7917
|
+
}
|
|
7918
|
+
const arrayBuffer = await response.arrayBuffer();
|
|
7919
|
+
const gltf2 = await new Promise((resolve2, reject) => {
|
|
7920
|
+
loader.parse(arrayBuffer, "", resolve2, reject);
|
|
7790
7921
|
});
|
|
7791
|
-
|
|
7792
|
-
this.geometry.setAttribute("position", position);
|
|
7793
|
-
this._timer = null;
|
|
7922
|
+
resolve(gltf2.scene);
|
|
7794
7923
|
});
|
|
7795
|
-
|
|
7796
|
-
|
|
7797
|
-
function selectLocalFileFun() {
|
|
7798
|
-
return new Promise((resolve) => {
|
|
7799
|
-
const input = document.createElement("input");
|
|
7800
|
-
input.type = "file";
|
|
7801
|
-
input.accept = "application/json";
|
|
7802
|
-
input.click();
|
|
7803
|
-
input.onchange = () => {
|
|
7804
|
-
if (input.files?.length) resolve(input.files[0]);
|
|
7805
|
-
else resolve(null);
|
|
7806
|
-
};
|
|
7807
|
-
});
|
|
7808
|
-
}
|
|
7809
|
-
const SelectLocalFile = Object.assign(selectLocalFileFun, {
|
|
7810
|
-
arrayBuffer() {
|
|
7811
|
-
return new Promise(async (resolve) => {
|
|
7812
|
-
const file = await selectLocalFileFun();
|
|
7813
|
-
if (file instanceof File) {
|
|
7814
|
-
const fileReader = new FileReader();
|
|
7815
|
-
fileReader.onload = () => {
|
|
7816
|
-
resolve(fileReader.result);
|
|
7817
|
-
};
|
|
7818
|
-
fileReader.readAsArrayBuffer(file);
|
|
7819
|
-
} else resolve(null);
|
|
7820
|
-
});
|
|
7821
|
-
},
|
|
7822
|
-
text() {
|
|
7823
|
-
return new Promise(async (resolve) => {
|
|
7824
|
-
const file = await selectLocalFileFun();
|
|
7825
|
-
if (file instanceof File) {
|
|
7826
|
-
const fileReader = new FileReader();
|
|
7827
|
-
fileReader.onload = () => {
|
|
7828
|
-
resolve(fileReader.result);
|
|
7829
|
-
};
|
|
7830
|
-
fileReader.readAsText(file, "utf-8");
|
|
7831
|
-
} else resolve(null);
|
|
7832
|
-
});
|
|
7833
|
-
},
|
|
7834
|
-
async json() {
|
|
7835
|
-
const text = await this.text();
|
|
7836
|
-
if (text) return JSON.parse(text);
|
|
7837
|
-
}
|
|
7838
|
-
});
|
|
7839
|
-
class CommandFlow extends EventDispatcher {
|
|
7840
|
-
list = [];
|
|
7841
|
-
rollbacklist = [];
|
|
7842
|
-
revokeRollbacklist = [];
|
|
7843
|
-
// 是否写入操作记录
|
|
7844
|
-
writeOperationList = true;
|
|
7845
|
-
loop = false;
|
|
7846
|
-
setLoop(loop) {
|
|
7847
|
-
this.loop = loop;
|
|
7848
|
-
return this;
|
|
7849
|
-
}
|
|
7850
|
-
/**
|
|
7851
|
-
*
|
|
7852
|
-
* @param operation
|
|
7853
|
-
* @returns
|
|
7854
|
-
*/
|
|
7855
|
-
add(operation) {
|
|
7856
|
-
this.list.push(operation);
|
|
7857
|
-
return this;
|
|
7858
|
-
}
|
|
7859
|
-
/** 添加回滚回调列表
|
|
7860
|
-
* @param callBack
|
|
7861
|
-
*/
|
|
7862
|
-
addRollback(callBack) {
|
|
7863
|
-
this.rollbacklist.push(callBack);
|
|
7864
|
-
return this;
|
|
7865
|
-
}
|
|
7866
|
-
/** 添加撤回回滚回调列表
|
|
7867
|
-
* @param callBack
|
|
7868
|
-
* @returns
|
|
7869
|
-
*/
|
|
7870
|
-
addRevokeRollback(callBack) {
|
|
7871
|
-
this.revokeRollbacklist.push(callBack);
|
|
7872
|
-
return this;
|
|
7873
|
-
}
|
|
7874
|
-
}
|
|
7875
|
-
class CommandManager extends EventDispatcher {
|
|
7876
|
-
commandFlowMap = /* @__PURE__ */ new Map();
|
|
7877
|
-
lock = false;
|
|
7878
|
-
abortController = null;
|
|
7879
|
-
resolve = null;
|
|
7880
|
-
currentName = null;
|
|
7881
|
-
_disabled = false;
|
|
7882
|
-
set disabled(disabled) {
|
|
7883
|
-
this._disabled = disabled;
|
|
7884
|
-
if (this._disabled) this.cancel();
|
|
7885
|
-
}
|
|
7886
|
-
get disabled() {
|
|
7887
|
-
return this._disabled;
|
|
7888
|
-
}
|
|
7889
|
-
/**
|
|
7890
|
-
* 操作记录
|
|
7891
|
-
*/
|
|
7892
|
-
operationList = [];
|
|
7893
|
-
rollbackList = [];
|
|
7894
|
-
constructor() {
|
|
7895
|
-
super();
|
|
7896
|
-
}
|
|
7897
|
-
/** 添加命令流
|
|
7898
|
-
* @param name
|
|
7899
|
-
* @returns
|
|
7900
|
-
*/
|
|
7901
|
-
addCommandFlow(name) {
|
|
7902
|
-
if (this.commandFlowMap.has(name)) throw new Error(`${name} 命令已经存在`);
|
|
7903
|
-
const commandFlow = new CommandFlow();
|
|
7904
|
-
this.commandFlowMap.set(name, commandFlow);
|
|
7905
|
-
return commandFlow;
|
|
7906
|
-
}
|
|
7907
|
-
executionPromise = null;
|
|
7908
|
-
executionResolve = null;
|
|
7909
|
-
/** 执行控制流
|
|
7910
|
-
* @param name
|
|
7911
|
-
* @returns
|
|
7912
|
-
*/
|
|
7913
|
-
async start(name, data = null, step = 0) {
|
|
7914
|
-
if (this.disabled) throw new Error("命令管理器已禁用,无法启动新的命令流");
|
|
7915
|
-
this.dispatchEvent({
|
|
7916
|
-
type: "startedBefore",
|
|
7917
|
-
name,
|
|
7918
|
-
currentName: this.currentName
|
|
7919
|
-
});
|
|
7920
|
-
this.executionPromise && await this.executionPromise;
|
|
7921
|
-
this.executionPromise = null;
|
|
7922
|
-
if (this.lock) {
|
|
7923
|
-
throw new Error("命令管理器已被 " + this.currentName + " 命令锁定,无法启动新的命令流,请退出或等待命令执行结束");
|
|
7924
|
-
}
|
|
7925
|
-
const commandFlow = this.commandFlowMap.get(name);
|
|
7926
|
-
if (!commandFlow) {
|
|
7927
|
-
throw new Error(`命令流 ${name} 不存在`);
|
|
7928
|
-
}
|
|
7929
|
-
this.lock = true;
|
|
7930
|
-
this.abortController = new AbortController();
|
|
7931
|
-
this.currentName = name;
|
|
7932
|
-
commandFlow.dispatchEvent({ type: "started" });
|
|
7933
|
-
this.dispatchEvent({ type: "started", name });
|
|
7934
|
-
try {
|
|
7935
|
-
for (let i = step; i < commandFlow.list.length; i++) {
|
|
7936
|
-
const operation = commandFlow.list[i];
|
|
7937
|
-
commandFlow.dispatchEvent({ type: "executing", index: i });
|
|
7938
|
-
this.dispatchEvent({ type: "executing", name, index: i });
|
|
7939
|
-
data = await new Promise((resolve) => {
|
|
7940
|
-
this.resolve = resolve;
|
|
7941
|
-
operation(resolve, data);
|
|
7942
|
-
});
|
|
7943
|
-
if (this.abortController.signal.aborted) {
|
|
7944
|
-
commandFlow.dispatchEvent({ type: "executionInterrupt", index: i });
|
|
7945
|
-
this.dispatchEvent({ type: "executionInterrupt", name, index: i });
|
|
7946
|
-
this.dispatchEvent({ type: "cancel", name });
|
|
7947
|
-
break;
|
|
7948
|
-
} else {
|
|
7949
|
-
commandFlow.dispatchEvent({ type: "executionCompleted", index: i, data });
|
|
7950
|
-
this.dispatchEvent({ type: "executionCompleted", name, index: i, data });
|
|
7951
|
-
}
|
|
7952
|
-
}
|
|
7953
|
-
} catch (error) {
|
|
7954
|
-
console.error(error);
|
|
7955
|
-
} finally {
|
|
7956
|
-
this.lock = false;
|
|
7957
|
-
this.currentName = null;
|
|
7958
|
-
if (this.abortController && !this.abortController.signal.aborted) {
|
|
7959
|
-
commandFlow.dispatchEvent({ type: "completed", data });
|
|
7960
|
-
this.dispatchEvent({ type: "completed", name, data });
|
|
7961
|
-
if (commandFlow.writeOperationList) {
|
|
7962
|
-
this.operationList.push({ name, data });
|
|
7963
|
-
this.rollbackList.length = 0;
|
|
7964
|
-
}
|
|
7965
|
-
if (commandFlow.loop) queueMicrotask(() => this.start(name));
|
|
7966
|
-
}
|
|
7967
|
-
this.abortController = null;
|
|
7968
|
-
commandFlow.dispatchEvent({ type: "finally" });
|
|
7969
|
-
this.dispatchEvent({ type: "finally", name });
|
|
7970
|
-
if (this.executionResolve) {
|
|
7971
|
-
this.executionResolve(null);
|
|
7972
|
-
this.executionResolve = null;
|
|
7973
|
-
}
|
|
7974
|
-
}
|
|
7975
|
-
return data;
|
|
7976
|
-
}
|
|
7977
|
-
/** 取消当前命令
|
|
7978
|
-
*/
|
|
7979
|
-
cancel() {
|
|
7980
|
-
if (this.abortController) {
|
|
7981
|
-
this.abortController.abort();
|
|
7982
|
-
if (this.resolve) this.resolve();
|
|
7983
|
-
this.executionPromise = new Promise((resolve) => this.executionResolve = resolve);
|
|
7984
|
-
}
|
|
7985
|
-
}
|
|
7986
|
-
/**
|
|
7987
|
-
* 回滚
|
|
7988
|
-
*/
|
|
7989
|
-
rollback() {
|
|
7990
|
-
try {
|
|
7991
|
-
const operation = this.operationList.pop();
|
|
7992
|
-
if (!operation) return false;
|
|
7993
|
-
const commandFlow = this.commandFlowMap.get(operation.name);
|
|
7994
|
-
if (!commandFlow) return false;
|
|
7995
|
-
const data = commandFlow.rollbacklist.reduce((data2, callBack) => callBack(data2), operation.data);
|
|
7996
|
-
this.dispatchEvent({ type: "rollback", name: operation.name });
|
|
7997
|
-
this.rollbackList.push({
|
|
7998
|
-
data,
|
|
7999
|
-
name: operation.name
|
|
8000
|
-
});
|
|
8001
|
-
return true;
|
|
8002
|
-
} catch (error) {
|
|
8003
|
-
throw new Error(`回滚失败:${error}`);
|
|
8004
|
-
}
|
|
8005
|
-
}
|
|
8006
|
-
/**
|
|
8007
|
-
* 撤销回滚
|
|
8008
|
-
*/
|
|
8009
|
-
revokeRollback() {
|
|
8010
|
-
try {
|
|
8011
|
-
const operation = this.rollbackList.pop();
|
|
8012
|
-
if (!operation) return false;
|
|
8013
|
-
const commandFlow = this.commandFlowMap.get(operation.name);
|
|
8014
|
-
if (!commandFlow) return false;
|
|
8015
|
-
const data = commandFlow.revokeRollbacklist.reduce((data2, callBack) => callBack(data2), operation.data);
|
|
8016
|
-
this.dispatchEvent({ type: "revokeRollback", name: operation.name });
|
|
8017
|
-
this.operationList.push({ name: operation.name, data });
|
|
8018
|
-
return true;
|
|
8019
|
-
} catch (error) {
|
|
8020
|
-
throw new Error(`撤回回滚失败:${error}`);
|
|
8021
|
-
}
|
|
8022
|
-
}
|
|
8023
|
-
destroy() {
|
|
8024
|
-
this.cancel();
|
|
8025
|
-
this.addEventListener("cancel", () => {
|
|
8026
|
-
this.commandFlowMap.clear();
|
|
8027
|
-
this.lock = false;
|
|
8028
|
-
this.abortController = null;
|
|
8029
|
-
this.resolve = null;
|
|
8030
|
-
this.executionResolve = null;
|
|
8031
|
-
this.executionPromise = null;
|
|
8032
|
-
this.currentName = null;
|
|
8033
|
-
this.operationList.length = 0;
|
|
8034
|
-
this.rollbackList.length = 0;
|
|
8035
|
-
}, { once: true });
|
|
8036
|
-
}
|
|
8037
|
-
}
|
|
8038
|
-
if (typeof globalThis !== "undefined" && typeof ProgressEvent === "undefined") {
|
|
8039
|
-
globalThis.ProgressEvent = class ProgressEvent extends EventTarget {
|
|
8040
|
-
type;
|
|
8041
|
-
lengthComputable;
|
|
8042
|
-
loaded;
|
|
8043
|
-
total;
|
|
8044
|
-
bubbles;
|
|
8045
|
-
cancelable;
|
|
8046
|
-
defaultPrevented;
|
|
8047
|
-
eventPhase;
|
|
8048
|
-
timeStamp;
|
|
8049
|
-
isTrusted;
|
|
8050
|
-
constructor(type, options = {}) {
|
|
8051
|
-
super();
|
|
8052
|
-
this.type = type;
|
|
8053
|
-
this.lengthComputable = options.lengthComputable || false;
|
|
8054
|
-
this.loaded = options.loaded || 0;
|
|
8055
|
-
this.total = options.total || 0;
|
|
8056
|
-
this.bubbles = false;
|
|
8057
|
-
this.cancelable = false;
|
|
8058
|
-
this.defaultPrevented = false;
|
|
8059
|
-
this.eventPhase = 0;
|
|
8060
|
-
this.timeStamp = Date.now();
|
|
8061
|
-
this.isTrusted = true;
|
|
8062
|
-
}
|
|
8063
|
-
preventDefault() {
|
|
8064
|
-
}
|
|
8065
|
-
stopPropagation() {
|
|
8066
|
-
}
|
|
8067
|
-
stopImmediatePropagation() {
|
|
8068
|
-
}
|
|
8069
|
-
};
|
|
8070
|
-
}
|
|
8071
|
-
let gltfLoader = GLTFExporter;
|
|
8072
|
-
let gltfExporter = GLTFLoader;
|
|
8073
|
-
let promise;
|
|
8074
|
-
function GltfInit() {
|
|
8075
|
-
if (promise) return promise;
|
|
8076
|
-
promise = new Promise(async (resolve) => {
|
|
8077
|
-
try {
|
|
8078
|
-
const { GLTFLoader: GLTFLoader2 } = await (typeof window !== "undefined" ? import("three/addons/loaders/GLTFLoader.js") : include("node-three-gltf", false));
|
|
8079
|
-
gltfLoader = GLTFLoader2;
|
|
8080
|
-
const { GLTFExporter: GLTFExporter2 } = await (typeof window !== "undefined" ? import("three/addons/exporters/GLTFExporter.js") : include("node-three-gltf", false));
|
|
8081
|
-
gltfExporter = GLTFExporter2;
|
|
8082
|
-
} catch (error) {
|
|
8083
|
-
console.error(error);
|
|
8084
|
-
}
|
|
8085
|
-
resolve(true);
|
|
8086
|
-
});
|
|
8087
|
-
return promise;
|
|
8088
|
-
}
|
|
8089
|
-
const gltf = {
|
|
8090
|
-
get GLTFLoader() {
|
|
8091
|
-
return gltfLoader;
|
|
8092
|
-
},
|
|
8093
|
-
get GLTFExporter() {
|
|
8094
|
-
return gltfExporter;
|
|
8095
|
-
}
|
|
8096
|
-
};
|
|
8097
|
-
const index$2 = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
|
|
8098
|
-
__proto__: null,
|
|
8099
|
-
ArrayMap,
|
|
8100
|
-
AxisAlignCorr,
|
|
8101
|
-
BoundExt,
|
|
8102
|
-
Box2,
|
|
8103
|
-
CommandFlow,
|
|
8104
|
-
CommandManager,
|
|
8105
|
-
Component,
|
|
8106
|
-
ComponentManager,
|
|
8107
|
-
CountMap,
|
|
8108
|
-
DoubleWallHelper,
|
|
8109
|
-
DxfSystem,
|
|
8110
|
-
EventDispatcher,
|
|
8111
|
-
GltfInit,
|
|
8112
|
-
LineGroupType,
|
|
8113
|
-
LineIndexGenerator,
|
|
8114
|
-
LineSegment,
|
|
8115
|
-
LineSegmentUndirectedGraph,
|
|
8116
|
-
Lines,
|
|
8117
|
-
MapEnhance,
|
|
8118
|
-
Point,
|
|
8119
|
-
PointVirtualGrid,
|
|
8120
|
-
Polygon,
|
|
8121
|
-
PvgList,
|
|
8122
|
-
Quadtree,
|
|
8123
|
-
Rectangle,
|
|
8124
|
-
SceneAutoGenerat,
|
|
8125
|
-
SelectLocalFile,
|
|
8126
|
-
SetMap,
|
|
8127
|
-
UndirectedGraph,
|
|
8128
|
-
UnionFindSet,
|
|
8129
|
-
buildBayWindowGroup,
|
|
8130
|
-
buildDoubleWallGroup,
|
|
8131
|
-
clippingDoubleWall,
|
|
8132
|
-
clippingLineUserData,
|
|
8133
|
-
cloneUserData,
|
|
8134
|
-
closedPathArea,
|
|
8135
|
-
components,
|
|
8136
|
-
createPointVirtualGrid,
|
|
8137
|
-
createQuadtree,
|
|
8138
|
-
findCircleEdges,
|
|
8139
|
-
findClosedPolygons,
|
|
8140
|
-
findDiscretePoint,
|
|
8141
|
-
findDiscretePointLine,
|
|
8142
|
-
findDiscretePointLine2,
|
|
8143
|
-
findLargestCircle,
|
|
8144
|
-
findRingEdges,
|
|
8145
|
-
findVerticalReference,
|
|
8146
|
-
gltf,
|
|
8147
|
-
lineDataToOriginalData,
|
|
8148
|
-
lineDataToThreeVJiaJson,
|
|
8149
|
-
lineSegmentClipping,
|
|
8150
|
-
mergeLineUserData,
|
|
8151
|
-
mergeSmallestCircle,
|
|
8152
|
-
mergeWindow,
|
|
8153
|
-
originalDataToLineData,
|
|
8154
|
-
recomputedWindow,
|
|
8155
|
-
recomputedWindowCenter,
|
|
8156
|
-
ringsDeduplication: ringsDeduplication$1,
|
|
8157
|
-
smallestCircle,
|
|
8158
|
-
toOriginalDataItem,
|
|
8159
|
-
tools,
|
|
8160
|
-
uuid
|
|
8161
|
-
}, Symbol.toStringTag, { value: "Module" }));
|
|
8162
|
-
let loader;
|
|
8163
|
-
class LoadModel {
|
|
8164
|
-
static _cache = /* @__PURE__ */ new Map();
|
|
8165
|
-
static _nameMap = /* @__PURE__ */ new Map();
|
|
8166
|
-
static addNameMap(mapName, name) {
|
|
8167
|
-
this._nameMap.set(mapName, name);
|
|
8168
|
-
}
|
|
8169
|
-
// 获取模型
|
|
8170
|
-
static loadGlb(name) {
|
|
8171
|
-
if (this._nameMap.has(name)) name = this._nameMap.get(name);
|
|
8172
|
-
if (this._cache.has(name)) return this._cache.get(name);
|
|
8173
|
-
if (!loader) loader = new gltf.GLTFLoader();
|
|
8174
|
-
const promise2 = new Promise(async (resolve) => {
|
|
8175
|
-
const response = await fetch(`${VITE_OSS_BASEURL}models/${name}.glb`);
|
|
8176
|
-
if (!response.ok) {
|
|
8177
|
-
resolve(null);
|
|
8178
|
-
return;
|
|
8179
|
-
}
|
|
8180
|
-
const arrayBuffer = await response.arrayBuffer();
|
|
8181
|
-
const gltf2 = await new Promise((resolve2, reject) => {
|
|
8182
|
-
loader.parse(arrayBuffer, "", resolve2, reject);
|
|
8183
|
-
});
|
|
8184
|
-
resolve(gltf2.scene);
|
|
8185
|
-
});
|
|
8186
|
-
this._cache.set(name, promise2);
|
|
8187
|
-
return promise2;
|
|
7924
|
+
this._cache.set(name, promise2);
|
|
7925
|
+
return promise2;
|
|
8188
7926
|
}
|
|
8189
7927
|
}
|
|
8190
7928
|
LoadModel.addNameMap("water cooler", "饮水机1");
|
|
@@ -8278,10 +8016,12 @@ class Scenario {
|
|
|
8278
8016
|
}
|
|
8279
8017
|
// 绘制图形
|
|
8280
8018
|
async drawGraphics(lines, z, trajectoryJson) {
|
|
8019
|
+
const doorLines = lines.filter((line) => line.userData.isDoor);
|
|
8020
|
+
lines = buildDoubleWallGroup(lines.filter((line) => !line.userData.isDoor), doorLines);
|
|
8021
|
+
lines.push(...doorLines);
|
|
8281
8022
|
await this.modelLoader();
|
|
8282
8023
|
await this.windowModelLoader();
|
|
8283
8024
|
this.height = z;
|
|
8284
|
-
console.log("this.height", this.height);
|
|
8285
8025
|
this.lines = lines;
|
|
8286
8026
|
if (this.group) {
|
|
8287
8027
|
this.scene.remove(this.group);
|
|
@@ -8370,7 +8110,7 @@ class Scenario {
|
|
|
8370
8110
|
});
|
|
8371
8111
|
let d = [];
|
|
8372
8112
|
for (const i2 in c) {
|
|
8373
|
-
d.push(new THREE.Vector3(c[i2].dian.x, c[i2].dian.y,
|
|
8113
|
+
d.push(new THREE.Vector3(c[i2].dian.x, c[i2].dian.y, c[i2].dian.z));
|
|
8374
8114
|
}
|
|
8375
8115
|
const seen = /* @__PURE__ */ new Set();
|
|
8376
8116
|
const E = d.filter((coord) => {
|
|
@@ -8669,10 +8409,10 @@ class Scenario {
|
|
|
8669
8409
|
let menModel = this.windowModely.clone();
|
|
8670
8410
|
if (data.center && distance) {
|
|
8671
8411
|
menModel.position.set(data.center.x, data.center.y, 0);
|
|
8672
|
-
const
|
|
8673
|
-
|
|
8412
|
+
const box32 = new THREE.Box3();
|
|
8413
|
+
box32.setFromObject(menModel);
|
|
8674
8414
|
let size = new THREE.Vector3();
|
|
8675
|
-
|
|
8415
|
+
box32.getSize(size);
|
|
8676
8416
|
menModel.rotation.x = Math.PI / 2;
|
|
8677
8417
|
menModel.rotation.y = angleRad;
|
|
8678
8418
|
menModel.scale.set(distance / size.x, height / size.y, wallWidth / size.z);
|
|
@@ -8686,12 +8426,11 @@ class Scenario {
|
|
|
8686
8426
|
installCabinet() {
|
|
8687
8427
|
let cabinetModel = this.CabinetModel.clone();
|
|
8688
8428
|
cabinetModel.position.set(5, 5, 0);
|
|
8689
|
-
const
|
|
8690
|
-
|
|
8429
|
+
const box32 = new THREE.Box3();
|
|
8430
|
+
box32.setFromObject(cabinetModel);
|
|
8691
8431
|
let size = new THREE.Vector3();
|
|
8692
|
-
|
|
8432
|
+
box32.getSize(size);
|
|
8693
8433
|
cabinetModel.rotation.x = Math.PI / 2;
|
|
8694
|
-
console.log("卡农看:", cabinetModel.geometry);
|
|
8695
8434
|
cabinetModel.scale.set(0.7 / size.x, 0.7 / size.y, 0.3 / size.z);
|
|
8696
8435
|
this.group.add(cabinetModel);
|
|
8697
8436
|
}
|
|
@@ -8699,10 +8438,10 @@ class Scenario {
|
|
|
8699
8438
|
installTable() {
|
|
8700
8439
|
let tableModel = this.tableModel.clone();
|
|
8701
8440
|
tableModel.position.set(5, 4, 0);
|
|
8702
|
-
const
|
|
8703
|
-
|
|
8441
|
+
const box32 = new THREE.Box3();
|
|
8442
|
+
box32.setFromObject(tableModel);
|
|
8704
8443
|
let size = new THREE.Vector3();
|
|
8705
|
-
|
|
8444
|
+
box32.getSize(size);
|
|
8706
8445
|
tableModel.rotation.x = Math.PI / 2;
|
|
8707
8446
|
tableModel.scale.set(0.7 / size.x, 0.75 / size.y, 1.3 / size.z);
|
|
8708
8447
|
this.group.add(tableModel);
|
|
@@ -8711,10 +8450,10 @@ class Scenario {
|
|
|
8711
8450
|
installchair() {
|
|
8712
8451
|
let tableModel = this.tableModel.clone();
|
|
8713
8452
|
tableModel.position.set(5, 4, 0);
|
|
8714
|
-
const
|
|
8715
|
-
|
|
8453
|
+
const box32 = new THREE.Box3();
|
|
8454
|
+
box32.setFromObject(tableModel);
|
|
8716
8455
|
let size = new THREE.Vector3();
|
|
8717
|
-
|
|
8456
|
+
box32.getSize(size);
|
|
8718
8457
|
tableModel.rotation.x = Math.PI / 2;
|
|
8719
8458
|
tableModel.scale.set(0.7 / size.x, 0.75 / size.y, 1.3 / size.z);
|
|
8720
8459
|
this.group.add(tableModel);
|
|
@@ -8813,10 +8552,10 @@ class Scenario {
|
|
|
8813
8552
|
let menModel = this.doorModely.clone();
|
|
8814
8553
|
if (center && distance) {
|
|
8815
8554
|
menModel.position.set(center.x, center.y, groundHeight);
|
|
8816
|
-
const
|
|
8817
|
-
|
|
8555
|
+
const box32 = new THREE.Box3();
|
|
8556
|
+
box32.setFromObject(menModel);
|
|
8818
8557
|
let size = new THREE.Vector3();
|
|
8819
|
-
|
|
8558
|
+
box32.getSize(size);
|
|
8820
8559
|
menModel.rotation.x = Math.PI / 2;
|
|
8821
8560
|
menModel.rotation.y = angleRad;
|
|
8822
8561
|
menModel.scale.set(distance / size.x, (Height ? Height : this.doorHeight) / size.y, DEFAULT_WALL_WIDTH / size.z);
|
|
@@ -9009,7 +8748,6 @@ class Scenario {
|
|
|
9009
8748
|
this.group.add(plane);
|
|
9010
8749
|
}
|
|
9011
8750
|
}
|
|
9012
|
-
const planeSrc = "";
|
|
9013
8751
|
function Rr(t, e, n = 0, r = t.length - 1, i = Mu) {
|
|
9014
8752
|
for (; r > n; ) {
|
|
9015
8753
|
if (r - n > 600) {
|
|
@@ -16050,514 +15788,13 @@ const { matrix: zr, transpose: ka, multiply: Ne } = ki({
|
|
|
16050
15788
|
return [
|
|
16051
15789
|
Ne([D, h], g),
|
|
16052
15790
|
Ne([M, h], g),
|
|
16053
|
-
Ne([M, w], g),
|
|
16054
|
-
Ne([D, w], g),
|
|
16055
|
-
Ne([D, h], g)
|
|
16056
|
-
];
|
|
16057
|
-
};
|
|
16058
|
-
class Capsule {
|
|
16059
|
-
/**
|
|
16060
|
-
* Constructs a new capsule.
|
|
16061
|
-
*
|
|
16062
|
-
* @param {Vector3} [start] - The start vector.
|
|
16063
|
-
* @param {Vector3} [end] - The end vector.
|
|
16064
|
-
* @param {number} [radius=1] - The capsule's radius.
|
|
16065
|
-
*/
|
|
16066
|
-
constructor(start = new Vector3(0, 0, 0), end = new Vector3(0, 1, 0), radius = 1) {
|
|
16067
|
-
this.start = start;
|
|
16068
|
-
this.end = end;
|
|
16069
|
-
this.radius = radius;
|
|
16070
|
-
}
|
|
16071
|
-
/**
|
|
16072
|
-
* Returns a new capsule with copied values from this instance.
|
|
16073
|
-
*
|
|
16074
|
-
* @return {Capsule} A clone of this instance.
|
|
16075
|
-
*/
|
|
16076
|
-
clone() {
|
|
16077
|
-
return new this.constructor().copy(this);
|
|
16078
|
-
}
|
|
16079
|
-
/**
|
|
16080
|
-
* Sets the capsule components to the given values.
|
|
16081
|
-
* Please note that this method only copies the values from the given objects.
|
|
16082
|
-
*
|
|
16083
|
-
* @param {Vector3} start - The start vector.
|
|
16084
|
-
* @param {Vector3} end - The end vector
|
|
16085
|
-
* @param {number} radius - The capsule's radius.
|
|
16086
|
-
* @return {Capsule} A reference to this capsule.
|
|
16087
|
-
*/
|
|
16088
|
-
set(start, end, radius) {
|
|
16089
|
-
this.start.copy(start);
|
|
16090
|
-
this.end.copy(end);
|
|
16091
|
-
this.radius = radius;
|
|
16092
|
-
return this;
|
|
16093
|
-
}
|
|
16094
|
-
/**
|
|
16095
|
-
* Copies the values of the given capsule to this instance.
|
|
16096
|
-
*
|
|
16097
|
-
* @param {Capsule} capsule - The capsule to copy.
|
|
16098
|
-
* @return {Capsule} A reference to this capsule.
|
|
16099
|
-
*/
|
|
16100
|
-
copy(capsule) {
|
|
16101
|
-
this.start.copy(capsule.start);
|
|
16102
|
-
this.end.copy(capsule.end);
|
|
16103
|
-
this.radius = capsule.radius;
|
|
16104
|
-
return this;
|
|
16105
|
-
}
|
|
16106
|
-
/**
|
|
16107
|
-
* Returns the center point of this capsule.
|
|
16108
|
-
*
|
|
16109
|
-
* @param {Vector3} target - The target vector that is used to store the method's result.
|
|
16110
|
-
* @return {Vector3} The center point.
|
|
16111
|
-
*/
|
|
16112
|
-
getCenter(target) {
|
|
16113
|
-
return target.copy(this.end).add(this.start).multiplyScalar(0.5);
|
|
16114
|
-
}
|
|
16115
|
-
/**
|
|
16116
|
-
* Adds the given offset to this capsule, effectively moving it in 3D space.
|
|
16117
|
-
*
|
|
16118
|
-
* @param {Vector3} v - The offset that should be used to translate the capsule.
|
|
16119
|
-
* @return {Capsule} A reference to this capsule.
|
|
16120
|
-
*/
|
|
16121
|
-
translate(v) {
|
|
16122
|
-
this.start.add(v);
|
|
16123
|
-
this.end.add(v);
|
|
16124
|
-
return this;
|
|
16125
|
-
}
|
|
16126
|
-
/**
|
|
16127
|
-
* Returns `true` if the given bounding box intersects with this capsule.
|
|
16128
|
-
*
|
|
16129
|
-
* @param {Box3} box - The bounding box to test.
|
|
16130
|
-
* @return {boolean} Whether the given bounding box intersects with this capsule.
|
|
16131
|
-
*/
|
|
16132
|
-
intersectsBox(box) {
|
|
16133
|
-
return checkAABBAxis(
|
|
16134
|
-
this.start.x,
|
|
16135
|
-
this.start.y,
|
|
16136
|
-
this.end.x,
|
|
16137
|
-
this.end.y,
|
|
16138
|
-
box.min.x,
|
|
16139
|
-
box.max.x,
|
|
16140
|
-
box.min.y,
|
|
16141
|
-
box.max.y,
|
|
16142
|
-
this.radius
|
|
16143
|
-
) && checkAABBAxis(
|
|
16144
|
-
this.start.x,
|
|
16145
|
-
this.start.z,
|
|
16146
|
-
this.end.x,
|
|
16147
|
-
this.end.z,
|
|
16148
|
-
box.min.x,
|
|
16149
|
-
box.max.x,
|
|
16150
|
-
box.min.z,
|
|
16151
|
-
box.max.z,
|
|
16152
|
-
this.radius
|
|
16153
|
-
) && checkAABBAxis(
|
|
16154
|
-
this.start.y,
|
|
16155
|
-
this.start.z,
|
|
16156
|
-
this.end.y,
|
|
16157
|
-
this.end.z,
|
|
16158
|
-
box.min.y,
|
|
16159
|
-
box.max.y,
|
|
16160
|
-
box.min.z,
|
|
16161
|
-
box.max.z,
|
|
16162
|
-
this.radius
|
|
16163
|
-
);
|
|
16164
|
-
}
|
|
16165
|
-
}
|
|
16166
|
-
function checkAABBAxis(p1x, p1y, p2x, p2y, minx, maxx, miny, maxy, radius) {
|
|
16167
|
-
return (minx - p1x < radius || minx - p2x < radius) && (p1x - maxx < radius || p2x - maxx < radius) && (miny - p1y < radius || miny - p2y < radius) && (p1y - maxy < radius || p2y - maxy < radius);
|
|
16168
|
-
}
|
|
16169
|
-
const _v1 = new Vector3();
|
|
16170
|
-
const _v2 = new Vector3();
|
|
16171
|
-
const _point1 = new Vector3();
|
|
16172
|
-
const _point2 = new Vector3();
|
|
16173
|
-
const _plane = new Plane();
|
|
16174
|
-
const _line1 = new Line3();
|
|
16175
|
-
const _line2 = new Line3();
|
|
16176
|
-
const _sphere = new Sphere();
|
|
16177
|
-
const _capsule = new Capsule();
|
|
16178
|
-
const _temp1 = new Vector3();
|
|
16179
|
-
const _temp2 = new Vector3();
|
|
16180
|
-
const _temp3 = new Vector3();
|
|
16181
|
-
const EPS = 1e-10;
|
|
16182
|
-
function lineToLineClosestPoints(line1, line2, target1 = null, target2 = null) {
|
|
16183
|
-
const r = _temp1.copy(line1.end).sub(line1.start);
|
|
16184
|
-
const s = _temp2.copy(line2.end).sub(line2.start);
|
|
16185
|
-
const w = _temp3.copy(line2.start).sub(line1.start);
|
|
16186
|
-
const a = r.dot(s), b = r.dot(r), c = s.dot(s), d = s.dot(w), e = r.dot(w);
|
|
16187
|
-
let t1, t2;
|
|
16188
|
-
const divisor = b * c - a * a;
|
|
16189
|
-
if (Math.abs(divisor) < EPS) {
|
|
16190
|
-
const d1 = -d / c;
|
|
16191
|
-
const d2 = (a - d) / c;
|
|
16192
|
-
if (Math.abs(d1 - 0.5) < Math.abs(d2 - 0.5)) {
|
|
16193
|
-
t1 = 0;
|
|
16194
|
-
t2 = d1;
|
|
16195
|
-
} else {
|
|
16196
|
-
t1 = 1;
|
|
16197
|
-
t2 = d2;
|
|
16198
|
-
}
|
|
16199
|
-
} else {
|
|
16200
|
-
t1 = (d * a + e * c) / divisor;
|
|
16201
|
-
t2 = (t1 * a - d) / c;
|
|
16202
|
-
}
|
|
16203
|
-
t2 = Math.max(0, Math.min(1, t2));
|
|
16204
|
-
t1 = Math.max(0, Math.min(1, t1));
|
|
16205
|
-
if (target1) {
|
|
16206
|
-
target1.copy(r).multiplyScalar(t1).add(line1.start);
|
|
16207
|
-
}
|
|
16208
|
-
if (target2) {
|
|
16209
|
-
target2.copy(s).multiplyScalar(t2).add(line2.start);
|
|
16210
|
-
}
|
|
16211
|
-
}
|
|
16212
|
-
class Octree {
|
|
16213
|
-
/**
|
|
16214
|
-
* Constructs a new Octree.
|
|
16215
|
-
*
|
|
16216
|
-
* @param {Box3} [box] - The base box with enclose the entire Octree.
|
|
16217
|
-
*/
|
|
16218
|
-
constructor(box) {
|
|
16219
|
-
this.box = box;
|
|
16220
|
-
this.bounds = new Box3();
|
|
16221
|
-
this.layers = new Layers();
|
|
16222
|
-
this.trianglesPerLeaf = 8;
|
|
16223
|
-
this.maxLevel = 16;
|
|
16224
|
-
this.subTrees = [];
|
|
16225
|
-
this.triangles = [];
|
|
16226
|
-
}
|
|
16227
|
-
/**
|
|
16228
|
-
* Adds the given triangle to the Octree. The triangle vertices are clamped if they exceed
|
|
16229
|
-
* the bounds of the Octree.
|
|
16230
|
-
*
|
|
16231
|
-
* @param {Triangle} triangle - The triangle to add.
|
|
16232
|
-
* @return {Octree} A reference to this Octree.
|
|
16233
|
-
*/
|
|
16234
|
-
addTriangle(triangle) {
|
|
16235
|
-
this.bounds.min.x = Math.min(this.bounds.min.x, triangle.a.x, triangle.b.x, triangle.c.x);
|
|
16236
|
-
this.bounds.min.y = Math.min(this.bounds.min.y, triangle.a.y, triangle.b.y, triangle.c.y);
|
|
16237
|
-
this.bounds.min.z = Math.min(this.bounds.min.z, triangle.a.z, triangle.b.z, triangle.c.z);
|
|
16238
|
-
this.bounds.max.x = Math.max(this.bounds.max.x, triangle.a.x, triangle.b.x, triangle.c.x);
|
|
16239
|
-
this.bounds.max.y = Math.max(this.bounds.max.y, triangle.a.y, triangle.b.y, triangle.c.y);
|
|
16240
|
-
this.bounds.max.z = Math.max(this.bounds.max.z, triangle.a.z, triangle.b.z, triangle.c.z);
|
|
16241
|
-
this.triangles.push(triangle);
|
|
16242
|
-
return this;
|
|
16243
|
-
}
|
|
16244
|
-
/**
|
|
16245
|
-
* Prepares {@link Octree#box} for the build.
|
|
16246
|
-
*
|
|
16247
|
-
* @return {Octree} A reference to this Octree.
|
|
16248
|
-
*/
|
|
16249
|
-
calcBox() {
|
|
16250
|
-
this.box = this.bounds.clone();
|
|
16251
|
-
this.box.min.x -= 0.01;
|
|
16252
|
-
this.box.min.y -= 0.01;
|
|
16253
|
-
this.box.min.z -= 0.01;
|
|
16254
|
-
return this;
|
|
16255
|
-
}
|
|
16256
|
-
/**
|
|
16257
|
-
* Splits the Octree. This method is used recursively when
|
|
16258
|
-
* building the Octree.
|
|
16259
|
-
*
|
|
16260
|
-
* @param {number} level - The current level.
|
|
16261
|
-
* @return {Octree} A reference to this Octree.
|
|
16262
|
-
*/
|
|
16263
|
-
split(level) {
|
|
16264
|
-
if (!this.box) return;
|
|
16265
|
-
const subTrees = [];
|
|
16266
|
-
const halfsize = _v2.copy(this.box.max).sub(this.box.min).multiplyScalar(0.5);
|
|
16267
|
-
for (let x = 0; x < 2; x++) {
|
|
16268
|
-
for (let y = 0; y < 2; y++) {
|
|
16269
|
-
for (let z = 0; z < 2; z++) {
|
|
16270
|
-
const box = new Box3();
|
|
16271
|
-
const v = _v1.set(x, y, z);
|
|
16272
|
-
box.min.copy(this.box.min).add(v.multiply(halfsize));
|
|
16273
|
-
box.max.copy(box.min).add(halfsize);
|
|
16274
|
-
subTrees.push(new Octree(box));
|
|
16275
|
-
}
|
|
16276
|
-
}
|
|
16277
|
-
}
|
|
16278
|
-
let triangle;
|
|
16279
|
-
while (triangle = this.triangles.pop()) {
|
|
16280
|
-
for (let i = 0; i < subTrees.length; i++) {
|
|
16281
|
-
if (subTrees[i].box.intersectsTriangle(triangle)) {
|
|
16282
|
-
subTrees[i].triangles.push(triangle);
|
|
16283
|
-
}
|
|
16284
|
-
}
|
|
16285
|
-
}
|
|
16286
|
-
for (let i = 0; i < subTrees.length; i++) {
|
|
16287
|
-
const len = subTrees[i].triangles.length;
|
|
16288
|
-
if (len > this.trianglesPerLeaf && level < this.maxLevel) {
|
|
16289
|
-
subTrees[i].split(level + 1);
|
|
16290
|
-
}
|
|
16291
|
-
if (len !== 0) {
|
|
16292
|
-
this.subTrees.push(subTrees[i]);
|
|
16293
|
-
}
|
|
16294
|
-
}
|
|
16295
|
-
return this;
|
|
16296
|
-
}
|
|
16297
|
-
/**
|
|
16298
|
-
* Builds the Octree.
|
|
16299
|
-
*
|
|
16300
|
-
* @return {Octree} A reference to this Octree.
|
|
16301
|
-
*/
|
|
16302
|
-
build() {
|
|
16303
|
-
this.calcBox();
|
|
16304
|
-
this.split(0);
|
|
16305
|
-
return this;
|
|
16306
|
-
}
|
|
16307
|
-
/**
|
|
16308
|
-
* Computes the triangles that potentially intersect with the given ray.
|
|
16309
|
-
*
|
|
16310
|
-
* @param {Ray} ray - The ray to test.
|
|
16311
|
-
* @param {Array<Triangle>} triangles - The target array that holds the triangles.
|
|
16312
|
-
*/
|
|
16313
|
-
getRayTriangles(ray, triangles) {
|
|
16314
|
-
for (let i = 0; i < this.subTrees.length; i++) {
|
|
16315
|
-
const subTree = this.subTrees[i];
|
|
16316
|
-
if (!ray.intersectsBox(subTree.box)) continue;
|
|
16317
|
-
if (subTree.triangles.length > 0) {
|
|
16318
|
-
for (let j = 0; j < subTree.triangles.length; j++) {
|
|
16319
|
-
if (triangles.indexOf(subTree.triangles[j]) === -1) triangles.push(subTree.triangles[j]);
|
|
16320
|
-
}
|
|
16321
|
-
} else {
|
|
16322
|
-
subTree.getRayTriangles(ray, triangles);
|
|
16323
|
-
}
|
|
16324
|
-
}
|
|
16325
|
-
}
|
|
16326
|
-
/**
|
|
16327
|
-
* Computes the intersection between the given capsule and triangle.
|
|
16328
|
-
*
|
|
16329
|
-
* @param {Capsule} capsule - The capsule to test.
|
|
16330
|
-
* @param {Triangle} triangle - The triangle to test.
|
|
16331
|
-
* @return {Object|false} The intersection object. If no intersection
|
|
16332
|
-
* is detected, the method returns `false`.
|
|
16333
|
-
*/
|
|
16334
|
-
triangleCapsuleIntersect(capsule, triangle) {
|
|
16335
|
-
triangle.getPlane(_plane);
|
|
16336
|
-
const d1 = _plane.distanceToPoint(capsule.start) - capsule.radius;
|
|
16337
|
-
const d2 = _plane.distanceToPoint(capsule.end) - capsule.radius;
|
|
16338
|
-
if (d1 > 0 && d2 > 0 || d1 < -capsule.radius && d2 < -capsule.radius) {
|
|
16339
|
-
return false;
|
|
16340
|
-
}
|
|
16341
|
-
const delta = Math.abs(d1 / (Math.abs(d1) + Math.abs(d2)));
|
|
16342
|
-
const intersectPoint = _v1.copy(capsule.start).lerp(capsule.end, delta);
|
|
16343
|
-
if (triangle.containsPoint(intersectPoint)) {
|
|
16344
|
-
return { normal: _plane.normal.clone(), point: intersectPoint.clone(), depth: Math.abs(Math.min(d1, d2)) };
|
|
16345
|
-
}
|
|
16346
|
-
const r2 = capsule.radius * capsule.radius;
|
|
16347
|
-
const line1 = _line1.set(capsule.start, capsule.end);
|
|
16348
|
-
const lines = [
|
|
16349
|
-
[triangle.a, triangle.b],
|
|
16350
|
-
[triangle.b, triangle.c],
|
|
16351
|
-
[triangle.c, triangle.a]
|
|
16352
|
-
];
|
|
16353
|
-
for (let i = 0; i < lines.length; i++) {
|
|
16354
|
-
const line2 = _line2.set(lines[i][0], lines[i][1]);
|
|
16355
|
-
lineToLineClosestPoints(line1, line2, _point1, _point2);
|
|
16356
|
-
if (_point1.distanceToSquared(_point2) < r2) {
|
|
16357
|
-
return {
|
|
16358
|
-
normal: _point1.clone().sub(_point2).normalize(),
|
|
16359
|
-
point: _point2.clone(),
|
|
16360
|
-
depth: capsule.radius - _point1.distanceTo(_point2)
|
|
16361
|
-
};
|
|
16362
|
-
}
|
|
16363
|
-
}
|
|
16364
|
-
return false;
|
|
16365
|
-
}
|
|
16366
|
-
/**
|
|
16367
|
-
* Computes the intersection between the given sphere and triangle.
|
|
16368
|
-
*
|
|
16369
|
-
* @param {Sphere} sphere - The sphere to test.
|
|
16370
|
-
* @param {Triangle} triangle - The triangle to test.
|
|
16371
|
-
* @return {Object|false} The intersection object. If no intersection
|
|
16372
|
-
* is detected, the method returns `false`.
|
|
16373
|
-
*/
|
|
16374
|
-
triangleSphereIntersect(sphere, triangle) {
|
|
16375
|
-
triangle.getPlane(_plane);
|
|
16376
|
-
if (!sphere.intersectsPlane(_plane)) return false;
|
|
16377
|
-
const depth = Math.abs(_plane.distanceToSphere(sphere));
|
|
16378
|
-
const r2 = sphere.radius * sphere.radius - depth * depth;
|
|
16379
|
-
const plainPoint = _plane.projectPoint(sphere.center, _v1);
|
|
16380
|
-
if (triangle.containsPoint(sphere.center)) {
|
|
16381
|
-
return { normal: _plane.normal.clone(), point: plainPoint.clone(), depth: Math.abs(_plane.distanceToSphere(sphere)) };
|
|
16382
|
-
}
|
|
16383
|
-
const lines = [
|
|
16384
|
-
[triangle.a, triangle.b],
|
|
16385
|
-
[triangle.b, triangle.c],
|
|
16386
|
-
[triangle.c, triangle.a]
|
|
16387
|
-
];
|
|
16388
|
-
for (let i = 0; i < lines.length; i++) {
|
|
16389
|
-
_line1.set(lines[i][0], lines[i][1]);
|
|
16390
|
-
_line1.closestPointToPoint(plainPoint, true, _v2);
|
|
16391
|
-
const d = _v2.distanceToSquared(sphere.center);
|
|
16392
|
-
if (d < r2) {
|
|
16393
|
-
return { normal: sphere.center.clone().sub(_v2).normalize(), point: _v2.clone(), depth: sphere.radius - Math.sqrt(d) };
|
|
16394
|
-
}
|
|
16395
|
-
}
|
|
16396
|
-
return false;
|
|
16397
|
-
}
|
|
16398
|
-
/**
|
|
16399
|
-
* Computes the triangles that potentially intersect with the given bounding sphere.
|
|
16400
|
-
*
|
|
16401
|
-
* @param {Sphere} sphere - The sphere to test.
|
|
16402
|
-
* @param {Array<Triangle>} triangles - The target array that holds the triangles.
|
|
16403
|
-
*/
|
|
16404
|
-
getSphereTriangles(sphere, triangles) {
|
|
16405
|
-
for (let i = 0; i < this.subTrees.length; i++) {
|
|
16406
|
-
const subTree = this.subTrees[i];
|
|
16407
|
-
if (!sphere.intersectsBox(subTree.box)) continue;
|
|
16408
|
-
if (subTree.triangles.length > 0) {
|
|
16409
|
-
for (let j = 0; j < subTree.triangles.length; j++) {
|
|
16410
|
-
if (triangles.indexOf(subTree.triangles[j]) === -1) triangles.push(subTree.triangles[j]);
|
|
16411
|
-
}
|
|
16412
|
-
} else {
|
|
16413
|
-
subTree.getSphereTriangles(sphere, triangles);
|
|
16414
|
-
}
|
|
16415
|
-
}
|
|
16416
|
-
}
|
|
16417
|
-
/**
|
|
16418
|
-
* Computes the triangles that potentially intersect with the given capsule.
|
|
16419
|
-
*
|
|
16420
|
-
* @param {Capsule} capsule - The capsule to test.
|
|
16421
|
-
* @param {Array<Triangle>} triangles - The target array that holds the triangles.
|
|
16422
|
-
*/
|
|
16423
|
-
getCapsuleTriangles(capsule, triangles) {
|
|
16424
|
-
for (let i = 0; i < this.subTrees.length; i++) {
|
|
16425
|
-
const subTree = this.subTrees[i];
|
|
16426
|
-
if (!capsule.intersectsBox(subTree.box)) continue;
|
|
16427
|
-
if (subTree.triangles.length > 0) {
|
|
16428
|
-
for (let j = 0; j < subTree.triangles.length; j++) {
|
|
16429
|
-
if (triangles.indexOf(subTree.triangles[j]) === -1) triangles.push(subTree.triangles[j]);
|
|
16430
|
-
}
|
|
16431
|
-
} else {
|
|
16432
|
-
subTree.getCapsuleTriangles(capsule, triangles);
|
|
16433
|
-
}
|
|
16434
|
-
}
|
|
16435
|
-
}
|
|
16436
|
-
/**
|
|
16437
|
-
* Performs a bounding sphere intersection test with this Octree.
|
|
16438
|
-
*
|
|
16439
|
-
* @param {Sphere} sphere - The bounding sphere to test.
|
|
16440
|
-
* @return {Object|boolean} The intersection object. If no intersection
|
|
16441
|
-
* is detected, the method returns `false`.
|
|
16442
|
-
*/
|
|
16443
|
-
sphereIntersect(sphere) {
|
|
16444
|
-
_sphere.copy(sphere);
|
|
16445
|
-
const triangles = [];
|
|
16446
|
-
let result, hit = false;
|
|
16447
|
-
this.getSphereTriangles(sphere, triangles);
|
|
16448
|
-
for (let i = 0; i < triangles.length; i++) {
|
|
16449
|
-
if (result = this.triangleSphereIntersect(_sphere, triangles[i])) {
|
|
16450
|
-
hit = true;
|
|
16451
|
-
_sphere.center.add(result.normal.multiplyScalar(result.depth));
|
|
16452
|
-
}
|
|
16453
|
-
}
|
|
16454
|
-
if (hit) {
|
|
16455
|
-
const collisionVector = _sphere.center.clone().sub(sphere.center);
|
|
16456
|
-
const depth = collisionVector.length();
|
|
16457
|
-
return { normal: collisionVector.normalize(), depth };
|
|
16458
|
-
}
|
|
16459
|
-
return false;
|
|
16460
|
-
}
|
|
16461
|
-
/**
|
|
16462
|
-
* Performs a capsule intersection test with this Octree.
|
|
16463
|
-
*
|
|
16464
|
-
* @param {Capsule} capsule - The capsule to test.
|
|
16465
|
-
* @return {Object|boolean} The intersection object. If no intersection
|
|
16466
|
-
* is detected, the method returns `false`.
|
|
16467
|
-
*/
|
|
16468
|
-
capsuleIntersect(capsule) {
|
|
16469
|
-
_capsule.copy(capsule);
|
|
16470
|
-
const triangles = [];
|
|
16471
|
-
let result, hit = false;
|
|
16472
|
-
this.getCapsuleTriangles(_capsule, triangles);
|
|
16473
|
-
for (let i = 0; i < triangles.length; i++) {
|
|
16474
|
-
if (result = this.triangleCapsuleIntersect(_capsule, triangles[i])) {
|
|
16475
|
-
hit = true;
|
|
16476
|
-
_capsule.translate(result.normal.multiplyScalar(result.depth));
|
|
16477
|
-
}
|
|
16478
|
-
}
|
|
16479
|
-
if (hit) {
|
|
16480
|
-
const collisionVector = _capsule.getCenter(new Vector3()).sub(capsule.getCenter(_v1));
|
|
16481
|
-
const depth = collisionVector.length();
|
|
16482
|
-
return { normal: collisionVector.normalize(), depth };
|
|
16483
|
-
}
|
|
16484
|
-
return false;
|
|
16485
|
-
}
|
|
16486
|
-
/**
|
|
16487
|
-
* Performs a ray intersection test with this Octree.
|
|
16488
|
-
*
|
|
16489
|
-
* @param {Ray} ray - The ray to test.
|
|
16490
|
-
* @return {Object|boolean} The nearest intersection object. If no intersection
|
|
16491
|
-
* is detected, the method returns `false`.
|
|
16492
|
-
*/
|
|
16493
|
-
rayIntersect(ray) {
|
|
16494
|
-
const triangles = [];
|
|
16495
|
-
let triangle, position, distance = 1e100;
|
|
16496
|
-
this.getRayTriangles(ray, triangles);
|
|
16497
|
-
for (let i = 0; i < triangles.length; i++) {
|
|
16498
|
-
const result = ray.intersectTriangle(triangles[i].a, triangles[i].b, triangles[i].c, true, _v1);
|
|
16499
|
-
if (result) {
|
|
16500
|
-
const newdistance = result.sub(ray.origin).length();
|
|
16501
|
-
if (distance > newdistance) {
|
|
16502
|
-
position = result.clone().add(ray.origin);
|
|
16503
|
-
distance = newdistance;
|
|
16504
|
-
triangle = triangles[i];
|
|
16505
|
-
}
|
|
16506
|
-
}
|
|
16507
|
-
}
|
|
16508
|
-
return distance < 1e100 ? { distance, triangle, position } : false;
|
|
16509
|
-
}
|
|
16510
|
-
/**
|
|
16511
|
-
* Constructs the Octree from the given 3D object.
|
|
16512
|
-
*
|
|
16513
|
-
* @param {Object3D} group - The scene graph node.
|
|
16514
|
-
* @return {Octree} A reference to this Octree.
|
|
16515
|
-
*/
|
|
16516
|
-
fromGraphNode(group2) {
|
|
16517
|
-
group2.updateWorldMatrix(true, true);
|
|
16518
|
-
group2.traverse((obj) => {
|
|
16519
|
-
if (obj.isMesh === true) {
|
|
16520
|
-
if (this.layers.test(obj.layers)) {
|
|
16521
|
-
let geometry, isTemp = false;
|
|
16522
|
-
if (obj.geometry.index !== null) {
|
|
16523
|
-
isTemp = true;
|
|
16524
|
-
geometry = obj.geometry.toNonIndexed();
|
|
16525
|
-
} else {
|
|
16526
|
-
geometry = obj.geometry;
|
|
16527
|
-
}
|
|
16528
|
-
const positionAttribute = geometry.getAttribute("position");
|
|
16529
|
-
for (let i = 0; i < positionAttribute.count; i += 3) {
|
|
16530
|
-
const v1 = new Vector3().fromBufferAttribute(positionAttribute, i);
|
|
16531
|
-
const v2 = new Vector3().fromBufferAttribute(positionAttribute, i + 1);
|
|
16532
|
-
const v3 = new Vector3().fromBufferAttribute(positionAttribute, i + 2);
|
|
16533
|
-
v1.applyMatrix4(obj.matrixWorld);
|
|
16534
|
-
v2.applyMatrix4(obj.matrixWorld);
|
|
16535
|
-
v3.applyMatrix4(obj.matrixWorld);
|
|
16536
|
-
this.addTriangle(new Triangle(v1, v2, v3));
|
|
16537
|
-
}
|
|
16538
|
-
if (isTemp) {
|
|
16539
|
-
geometry.dispose();
|
|
16540
|
-
}
|
|
16541
|
-
}
|
|
16542
|
-
}
|
|
16543
|
-
});
|
|
16544
|
-
this.build();
|
|
16545
|
-
return this;
|
|
16546
|
-
}
|
|
16547
|
-
/**
|
|
16548
|
-
* Clears the Octree by making it empty.
|
|
16549
|
-
*
|
|
16550
|
-
* @return {Octree} A reference to this Octree.
|
|
16551
|
-
*/
|
|
16552
|
-
clear() {
|
|
16553
|
-
this.box = null;
|
|
16554
|
-
this.bounds.makeEmpty();
|
|
16555
|
-
this.subTrees.length = 0;
|
|
16556
|
-
this.triangles.length = 0;
|
|
16557
|
-
return this;
|
|
16558
|
-
}
|
|
16559
|
-
}
|
|
15791
|
+
Ne([M, w], g),
|
|
15792
|
+
Ne([D, w], g),
|
|
15793
|
+
Ne([D, h], g)
|
|
15794
|
+
];
|
|
15795
|
+
};
|
|
16560
15796
|
let globalScenario;
|
|
15797
|
+
const box3 = new THREE.Box3();
|
|
16561
15798
|
class SceneAutoGenerat {
|
|
16562
15799
|
lines;
|
|
16563
15800
|
itemList;
|
|
@@ -16579,30 +15816,6 @@ class SceneAutoGenerat {
|
|
|
16579
15816
|
await this.buildItem();
|
|
16580
15817
|
globalScenario.scene.add(this.scene);
|
|
16581
15818
|
}
|
|
16582
|
-
/**
|
|
16583
|
-
* 获取包围矩形
|
|
16584
|
-
*/
|
|
16585
|
-
getSurroundRect(path, angle = this.angle) {
|
|
16586
|
-
const points = path.map((p) => Point.from(p)), box2 = Box2.fromByPoints(...points), center = box2.center;
|
|
16587
|
-
drawLines(points);
|
|
16588
|
-
points.forEach((p) => p.rotate(center, angle));
|
|
16589
|
-
const rectangle = Box2.fromByPoints(...points).points;
|
|
16590
|
-
rectangle.forEach((p) => p.rotate(center, -angle));
|
|
16591
|
-
return new Polygon(rectangle);
|
|
16592
|
-
}
|
|
16593
|
-
/** 构建平面
|
|
16594
|
-
*/
|
|
16595
|
-
buildPlane() {
|
|
16596
|
-
const texture = new THREE.TextureLoader().load(planeSrc);
|
|
16597
|
-
texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
|
|
16598
|
-
texture.repeat.set(10, 10);
|
|
16599
|
-
const plane = new THREE.Mesh(new THREE.BoxGeometry(50, 50, 1e-3), new THREE.MeshBasicMaterial({
|
|
16600
|
-
color: 13421772,
|
|
16601
|
-
map: texture
|
|
16602
|
-
}));
|
|
16603
|
-
plane.position.z = this.z;
|
|
16604
|
-
this.scene.add(plane);
|
|
16605
|
-
}
|
|
16606
15819
|
wallGroup = null;
|
|
16607
15820
|
/** 构建墙壁
|
|
16608
15821
|
*/
|
|
@@ -16610,37 +15823,13 @@ class SceneAutoGenerat {
|
|
|
16610
15823
|
this.wallGroup = await globalScenario.drawGraphics(this.lines, this.z, this.trajectory);
|
|
16611
15824
|
this.scene.add(this.wallGroup);
|
|
16612
15825
|
}
|
|
16613
|
-
/** 吸附
|
|
16614
|
-
* @param walls
|
|
16615
|
-
* @param group
|
|
16616
|
-
* @param proximityThreshold
|
|
16617
|
-
*/
|
|
16618
|
-
adsorption(octree, rectangle, height, proximityThreshold = 0.05) {
|
|
16619
|
-
const shape = new THREE.Shape();
|
|
16620
|
-
rectangle.forEach((p, i) => {
|
|
16621
|
-
if (i === 0) shape.moveTo(p.x, p.y);
|
|
16622
|
-
else shape.lineTo(p.x, p.y);
|
|
16623
|
-
if (i === rectangle.length - 1) shape.lineTo(rectangle[0].x, rectangle[0].y);
|
|
16624
|
-
});
|
|
16625
|
-
const geometry = new THREE.ExtrudeGeometry(shape, {
|
|
16626
|
-
depth: height,
|
|
16627
|
-
bevelEnabled: false
|
|
16628
|
-
});
|
|
16629
|
-
const material = new THREE.MeshBasicMaterial({ color: THREE.MathUtils.randInt(0, 16777215), opacity: 1, transparent: true });
|
|
16630
|
-
const mesh = new THREE.Mesh(geometry, material);
|
|
16631
|
-
mesh.position.z = this.z;
|
|
16632
|
-
this.scene.add(mesh);
|
|
16633
|
-
}
|
|
16634
15826
|
/** 获取模型
|
|
16635
15827
|
* @param item
|
|
16636
15828
|
* @param walls
|
|
16637
15829
|
*/
|
|
16638
|
-
async getModel(item
|
|
15830
|
+
async getModel(item) {
|
|
16639
15831
|
let model = await LoadModel.loadGlb(item.category);
|
|
16640
15832
|
if (!model) return;
|
|
16641
|
-
const contour = Point.fromByList(item.contour), rectangle = new Polygon(Qa(contour.map((p) => [p.x, p.y])).map((p) => Point.from(p)));
|
|
16642
|
-
rectangle.pop();
|
|
16643
|
-
const max = rectangle.getMaxLengthInfo(), min = rectangle.getMinLengthInfo(), center = rectangle.getCenter(), direction = max.start.y < max.end.y ? max.start.direction(max.end) : max.end.direction(max.start), height = Math.abs(item.box.max.z - this.z);
|
|
16644
15833
|
model = model.clone(true);
|
|
16645
15834
|
model.rotateX(Math.PI * 0.5);
|
|
16646
15835
|
model.traverse((mesh) => {
|
|
@@ -16650,388 +15839,759 @@ class SceneAutoGenerat {
|
|
|
16650
15839
|
material.vertexColors = false;
|
|
16651
15840
|
material.color = new THREE.Color(16777215);
|
|
16652
15841
|
}
|
|
16653
|
-
});
|
|
16654
|
-
const
|
|
16655
|
-
|
|
16656
|
-
|
|
16657
|
-
|
|
16658
|
-
|
|
16659
|
-
|
|
16660
|
-
|
|
16661
|
-
|
|
16662
|
-
group2
|
|
16663
|
-
|
|
16664
|
-
|
|
16665
|
-
|
|
16666
|
-
|
|
16667
|
-
|
|
16668
|
-
|
|
16669
|
-
|
|
16670
|
-
|
|
16671
|
-
point1.set(
|
|
16672
|
-
point2.set(
|
|
15842
|
+
});
|
|
15843
|
+
const contour = Point.fromByList(item.contour ?? []), rectangle = new Polygon(Qa(contour.map((p) => [p.x, p.y])).map((p) => Point.from(p)));
|
|
15844
|
+
rectangle.pop();
|
|
15845
|
+
const max = rectangle.getMaxLengthInfo(), min = rectangle.getMinLengthInfo(), center = rectangle.getCenter(), direction = max.start.y < max.end.y ? max.start.direction(max.end) : max.end.direction(max.start), height = Math.abs(item.box.max.z - this.z);
|
|
15846
|
+
const group2 = new THREE.Group();
|
|
15847
|
+
group2.position.set(center.x, center.y, this.z);
|
|
15848
|
+
group2.add(model);
|
|
15849
|
+
this.scene.add(group2);
|
|
15850
|
+
const box32 = new THREE.Box3();
|
|
15851
|
+
box32.setFromObject(group2);
|
|
15852
|
+
const size = box32.getSize(new THREE.Vector3()), maxKey = size.x > size.y ? "x" : "y", mimKey = size.x < size.y ? "x" : "y", maxS = max.length / size[maxKey], minS = min.length / size[mimKey], scale = height / size.z;
|
|
15853
|
+
group2.scale[maxKey] *= maxS;
|
|
15854
|
+
group2.scale[mimKey] *= minS;
|
|
15855
|
+
group2.scale.z *= scale;
|
|
15856
|
+
box32.setFromObject(group2);
|
|
15857
|
+
box32.getSize(size);
|
|
15858
|
+
const point1 = new Point(), point2 = new Point();
|
|
15859
|
+
if (size.x > size.y) {
|
|
15860
|
+
point1.set(box32.min.x, box32.min.y);
|
|
15861
|
+
point2.set(box32.max.x, box32.min.y);
|
|
15862
|
+
} else {
|
|
15863
|
+
point1.set(box32.min.x, box32.min.y);
|
|
15864
|
+
point2.set(box32.min.x, box32.max.y);
|
|
15865
|
+
}
|
|
15866
|
+
const direction2 = point1.y < point2.y ? point1.direction(point2) : point2.direction(point1);
|
|
15867
|
+
group2.rotateZ(direction2.angleBetween2(direction));
|
|
15868
|
+
this.scene.add(
|
|
15869
|
+
SceneAutoGenerat.itemParse(item, this.z).box
|
|
15870
|
+
);
|
|
15871
|
+
}
|
|
15872
|
+
/** 构建物品模型
|
|
15873
|
+
*/
|
|
15874
|
+
async buildItem() {
|
|
15875
|
+
await Promise.all(this.itemList.map(async (item) => await this.getModel(item)));
|
|
15876
|
+
}
|
|
15877
|
+
static itemParse(item, z) {
|
|
15878
|
+
const contour = Point.fromByList(item.contour), rectangle = new Polygon(Qa(contour.map((p) => [p.x, p.y])).map((p) => Point.from(p)));
|
|
15879
|
+
rectangle.pop();
|
|
15880
|
+
const height = Math.abs(item.box.max.z - z), max = rectangle.getMaxLengthInfo(), direction = max.start.y < max.end.y ? max.start.direction(max.end) : max.end.direction(max.start), shape = new THREE.Shape();
|
|
15881
|
+
rectangle.forEach((p, i) => {
|
|
15882
|
+
if (i === 0) shape.moveTo(p.x, p.y);
|
|
15883
|
+
else shape.lineTo(p.x, p.y);
|
|
15884
|
+
if (i === rectangle.length - 1) shape.lineTo(rectangle[0].x, rectangle[0].y);
|
|
15885
|
+
});
|
|
15886
|
+
const geometry = new THREE.ExtrudeGeometry(shape, {
|
|
15887
|
+
depth: height,
|
|
15888
|
+
bevelEnabled: false
|
|
15889
|
+
}), edges = new THREE.EdgesGeometry(geometry), lineSegments = new THREE.LineSegments(edges, new THREE.LineBasicMaterial({ color: 16776960 }));
|
|
15890
|
+
lineSegments.position.z = z;
|
|
15891
|
+
box3.setFromObject(lineSegments);
|
|
15892
|
+
const center = box3.getCenter(new THREE.Vector3());
|
|
15893
|
+
return {
|
|
15894
|
+
box: lineSegments,
|
|
15895
|
+
center,
|
|
15896
|
+
category: item.category,
|
|
15897
|
+
angle: direction.angleBetween(new Point(0, 1), "angle", "360") * (Math.PI / 180) + Math.PI * 0.5
|
|
15898
|
+
};
|
|
15899
|
+
}
|
|
15900
|
+
}
|
|
15901
|
+
const exporter = new OBJExporter();
|
|
15902
|
+
let glbExporter;
|
|
15903
|
+
function lineSqueezing(p1, p2, width = 0.1) {
|
|
15904
|
+
const normal = p2.normal(p1);
|
|
15905
|
+
const pDirect = p2.direction(p1).mutiplyScalar(width * 0.5);
|
|
15906
|
+
const nDirect = p1.direction(p2).mutiplyScalar(width * 0.5);
|
|
15907
|
+
const offsetX = normal.x * width * 0.5;
|
|
15908
|
+
const offsetY = normal.y * width * 0.5;
|
|
15909
|
+
return {
|
|
15910
|
+
points: [
|
|
15911
|
+
// 第一条线
|
|
15912
|
+
new Point(p1.x + offsetX, p1.y + offsetY).add(nDirect),
|
|
15913
|
+
new Point(p2.x + offsetX, p2.y + offsetY).add(pDirect),
|
|
15914
|
+
// 第二条线
|
|
15915
|
+
new Point(p1.x - offsetX, p1.y - offsetY).add(nDirect),
|
|
15916
|
+
new Point(p2.x - offsetX, p2.y - offsetY).add(pDirect)
|
|
15917
|
+
],
|
|
15918
|
+
indices: [0, 1, 1, 3, 3, 2, 2, 0],
|
|
15919
|
+
rectIndices: [0, 1, 3, 2, 0]
|
|
15920
|
+
};
|
|
15921
|
+
}
|
|
15922
|
+
class WhiteModel extends Component {
|
|
15923
|
+
static name = "WhiteModel";
|
|
15924
|
+
Dxf = null;
|
|
15925
|
+
Variable = null;
|
|
15926
|
+
// dxf数据白模
|
|
15927
|
+
whiteModelGroup = new THREE.Group();
|
|
15928
|
+
// dxf数据白模边缘线
|
|
15929
|
+
whiteModelLineGroup = new THREE.Group();
|
|
15930
|
+
// 原始数据白模
|
|
15931
|
+
originalWhiteMode = new THREE.Group();
|
|
15932
|
+
material = new THREE.MeshStandardMaterial({ color: 16777215, transparent: true, opacity: 0.8, side: THREE.DoubleSide });
|
|
15933
|
+
itemList = [];
|
|
15934
|
+
promise;
|
|
15935
|
+
/** 设置物品列表
|
|
15936
|
+
* @param itemList
|
|
15937
|
+
*/
|
|
15938
|
+
setItemList(itemList) {
|
|
15939
|
+
if (Array.isArray(itemList)) this.itemList = itemList;
|
|
15940
|
+
else if ("itemInfo" in itemList) this.itemList = itemList.itemInfo;
|
|
15941
|
+
}
|
|
15942
|
+
onAddFromParent(parent) {
|
|
15943
|
+
this.Dxf = parent.findComponentByName("Dxf");
|
|
15944
|
+
this.Variable = parent.findComponentByName("Variable");
|
|
15945
|
+
this.originalWhiteMode.visible = false;
|
|
15946
|
+
this.Dxf?.addEventListener("lineOffset", () => {
|
|
15947
|
+
this.promise = this.updateModel();
|
|
15948
|
+
});
|
|
15949
|
+
}
|
|
15950
|
+
async updateModel() {
|
|
15951
|
+
const dxfSystem = this.parent;
|
|
15952
|
+
this.Variable?.set("whiteModelVisible", false);
|
|
15953
|
+
const dxf = this.Dxf;
|
|
15954
|
+
this.originalWhiteMode.clear();
|
|
15955
|
+
this.whiteModelGroup.clear();
|
|
15956
|
+
this.whiteModelLineGroup.clear();
|
|
15957
|
+
this.whiteModelGroup.add(this.whiteModelLineGroup);
|
|
15958
|
+
this.whiteModelGroup.position.z = dxf.originalZAverage;
|
|
15959
|
+
this.originalWhiteMode.position.z = dxf.originalZAverage;
|
|
15960
|
+
const lines = dxfSystem.Dxf.getLineSegments(true);
|
|
15961
|
+
const sceneAutoGenerat = new SceneAutoGenerat(lines, this.itemList, dxfSystem.Dxf.originalZAverage, dxfSystem.CorrectionDxf.trajectory);
|
|
15962
|
+
await sceneAutoGenerat.init();
|
|
15963
|
+
const group2 = sceneAutoGenerat.scene;
|
|
15964
|
+
if (group2) {
|
|
15965
|
+
this.whiteModelLineGroup.add(group2.clone(true));
|
|
15966
|
+
}
|
|
15967
|
+
const walls = dxf.originalData.map(({ start, end, insetionArr }) => {
|
|
15968
|
+
const startVec3 = new Point(start.x, start.y).mutiplyScalar(dxf.scale), endVec3 = new Point(end.x, end.y).mutiplyScalar(dxf.scale), { points, indices, rectIndices } = lineSqueezing(startVec3, endVec3, dxf.width);
|
|
15969
|
+
return {
|
|
15970
|
+
points,
|
|
15971
|
+
indices,
|
|
15972
|
+
rectIndices,
|
|
15973
|
+
insetions: (insetionArr ?? []).map((insetion) => insetion.index)
|
|
15974
|
+
};
|
|
15975
|
+
});
|
|
15976
|
+
walls.forEach((wall) => {
|
|
15977
|
+
const shape = new THREE.Shape();
|
|
15978
|
+
wall.rectIndices.forEach((index2, i) => {
|
|
15979
|
+
const p = wall.points[index2];
|
|
15980
|
+
if (i === 0) shape.moveTo(p.x, p.y);
|
|
15981
|
+
else shape.lineTo(p.x, p.y);
|
|
15982
|
+
});
|
|
15983
|
+
const geometry = new THREE.ExtrudeGeometry(shape, {
|
|
15984
|
+
depth: 2.8,
|
|
15985
|
+
bevelSize: 0
|
|
15986
|
+
});
|
|
15987
|
+
if (geometry.attributes.position.array.filter((num) => Number.isNaN(num)).length) return;
|
|
15988
|
+
const mesh = new THREE.Mesh(geometry);
|
|
15989
|
+
this.originalWhiteMode?.add(mesh);
|
|
15990
|
+
});
|
|
15991
|
+
this.dispatchEvent({
|
|
15992
|
+
type: "updateModel",
|
|
15993
|
+
originalWhiteMode: this.originalWhiteMode,
|
|
15994
|
+
whiteModelGroup: this.whiteModelGroup
|
|
15995
|
+
});
|
|
15996
|
+
return this.whiteModelLineGroup;
|
|
15997
|
+
}
|
|
15998
|
+
/**
|
|
15999
|
+
* 转为obj
|
|
16000
|
+
* @returns
|
|
16001
|
+
*/
|
|
16002
|
+
toOBJ() {
|
|
16003
|
+
return new Promise(async (resolve) => {
|
|
16004
|
+
await this.promise;
|
|
16005
|
+
this.material.opacity = 1;
|
|
16006
|
+
this.material.needsUpdate = true;
|
|
16007
|
+
setTimeout(() => {
|
|
16008
|
+
resolve(exporter.parse(this.whiteModelGroup));
|
|
16009
|
+
this.material.opacity = 0.8;
|
|
16010
|
+
this.material.transparent = true;
|
|
16011
|
+
}, 20);
|
|
16012
|
+
});
|
|
16013
|
+
}
|
|
16014
|
+
/**
|
|
16015
|
+
* 转为 glb
|
|
16016
|
+
* @param binary
|
|
16017
|
+
* @returns
|
|
16018
|
+
*/
|
|
16019
|
+
toGltf(binary = true) {
|
|
16020
|
+
if (!glbExporter) glbExporter = new gltf.GLTFExporter();
|
|
16021
|
+
return new Promise(async (resolve) => {
|
|
16022
|
+
await this.promise;
|
|
16023
|
+
this.material.opacity = 1;
|
|
16024
|
+
this.material.needsUpdate = true;
|
|
16025
|
+
setTimeout(async () => {
|
|
16026
|
+
glbExporter.parse(this.whiteModelGroup.children, (gltf2) => {
|
|
16027
|
+
resolve(gltf2);
|
|
16028
|
+
this.material.opacity = 0.8;
|
|
16029
|
+
this.material.transparent = true;
|
|
16030
|
+
}, () => {
|
|
16031
|
+
resolve(void 0);
|
|
16032
|
+
}, {
|
|
16033
|
+
binary
|
|
16034
|
+
});
|
|
16035
|
+
}, 20);
|
|
16036
|
+
});
|
|
16037
|
+
}
|
|
16038
|
+
/**
|
|
16039
|
+
* 转为 OBJBlob
|
|
16040
|
+
* @returns
|
|
16041
|
+
*/
|
|
16042
|
+
async toOBJBlob() {
|
|
16043
|
+
const buffer = await this.toOBJ();
|
|
16044
|
+
if (buffer) {
|
|
16045
|
+
return new Blob([buffer], { type: "application/octet-stream" });
|
|
16046
|
+
}
|
|
16047
|
+
}
|
|
16048
|
+
/**
|
|
16049
|
+
* 转为 GltfBlob
|
|
16050
|
+
* @returns
|
|
16051
|
+
*/
|
|
16052
|
+
async toGltfBlob(binary = true) {
|
|
16053
|
+
const buffer = await this.toGltf(binary);
|
|
16054
|
+
if (buffer) {
|
|
16055
|
+
if (binary) return new Blob([buffer], { type: "application/octet-stream" });
|
|
16056
|
+
else return new Blob([JSON.stringify(buffer)], { type: "application/json" });
|
|
16057
|
+
}
|
|
16058
|
+
}
|
|
16059
|
+
/**
|
|
16060
|
+
* 下载 OBJ
|
|
16061
|
+
* @param filename
|
|
16062
|
+
* @returns
|
|
16063
|
+
*/
|
|
16064
|
+
async downloadOBJ(filename) {
|
|
16065
|
+
if (typeof window !== "undefined") {
|
|
16066
|
+
const blob = await this.toOBJBlob();
|
|
16067
|
+
if (!blob) return;
|
|
16068
|
+
const a = document.createElement("a");
|
|
16069
|
+
a.href = URL.createObjectURL(blob);
|
|
16070
|
+
a.download = filename;
|
|
16071
|
+
a.click();
|
|
16072
|
+
} else if (typeof global !== "undefined") {
|
|
16073
|
+
const buffer = await this.toOBJ();
|
|
16074
|
+
if (buffer) {
|
|
16075
|
+
const fs2 = await include("fs", false);
|
|
16076
|
+
fs2.writeFileSync(filename, buffer);
|
|
16077
|
+
}
|
|
16078
|
+
}
|
|
16079
|
+
}
|
|
16080
|
+
/**
|
|
16081
|
+
* 下载 Gltf
|
|
16082
|
+
* @param filename
|
|
16083
|
+
* @returns
|
|
16084
|
+
*/
|
|
16085
|
+
async downloadGltf(filename, binary = true) {
|
|
16086
|
+
if (typeof window !== "undefined") {
|
|
16087
|
+
const blob = await this.toGltfBlob(binary);
|
|
16088
|
+
if (!blob) return;
|
|
16089
|
+
const a = document.createElement("a");
|
|
16090
|
+
a.href = URL.createObjectURL(blob);
|
|
16091
|
+
a.download = filename;
|
|
16092
|
+
a.click();
|
|
16093
|
+
} else if (typeof global !== "undefined") {
|
|
16094
|
+
const buffer = await this.toGltf(binary);
|
|
16095
|
+
if (buffer) {
|
|
16096
|
+
const fs2 = await include("fs", false);
|
|
16097
|
+
fs2.writeFileSync(filename, binary ? Buffer.from(buffer) : JSON.stringify(buffer));
|
|
16098
|
+
}
|
|
16673
16099
|
}
|
|
16674
|
-
const direction2 = point1.y < point2.y ? point1.direction(point2) : point2.direction(point1);
|
|
16675
|
-
group2.rotateZ(direction2.angleBetween2(direction));
|
|
16676
|
-
}
|
|
16677
|
-
/** 构建物品模型
|
|
16678
|
-
*/
|
|
16679
|
-
async buildItem() {
|
|
16680
|
-
const octree = new Octree();
|
|
16681
|
-
octree.fromGraphNode(this.wallGroup);
|
|
16682
|
-
await Promise.all(this.itemList.map(async (item) => await this.getModel(item, octree)));
|
|
16683
16100
|
}
|
|
16684
16101
|
}
|
|
16685
|
-
|
|
16686
|
-
|
|
16687
|
-
function lineSqueezing(p1, p2, width = 0.1) {
|
|
16688
|
-
const normal = p2.normal(p1);
|
|
16689
|
-
const pDirect = p2.direction(p1).mutiplyScalar(width * 0.5);
|
|
16690
|
-
const nDirect = p1.direction(p2).mutiplyScalar(width * 0.5);
|
|
16691
|
-
const offsetX = normal.x * width * 0.5;
|
|
16692
|
-
const offsetY = normal.y * width * 0.5;
|
|
16693
|
-
return {
|
|
16694
|
-
points: [
|
|
16695
|
-
// 第一条线
|
|
16696
|
-
new Point(p1.x + offsetX, p1.y + offsetY).add(nDirect),
|
|
16697
|
-
new Point(p2.x + offsetX, p2.y + offsetY).add(pDirect),
|
|
16698
|
-
// 第二条线
|
|
16699
|
-
new Point(p1.x - offsetX, p1.y - offsetY).add(nDirect),
|
|
16700
|
-
new Point(p2.x - offsetX, p2.y - offsetY).add(pDirect)
|
|
16701
|
-
],
|
|
16702
|
-
indices: [0, 1, 1, 3, 3, 2, 2, 0],
|
|
16703
|
-
rectIndices: [0, 1, 3, 2, 0]
|
|
16704
|
-
};
|
|
16705
|
-
}
|
|
16706
|
-
class WhiteModel extends Component {
|
|
16707
|
-
static name = "WhiteModel";
|
|
16102
|
+
class DetailsPoint extends Component {
|
|
16103
|
+
static name = "DetailsPoint";
|
|
16708
16104
|
Dxf = null;
|
|
16105
|
+
WhiteModel = null;
|
|
16709
16106
|
Variable = null;
|
|
16710
|
-
|
|
16711
|
-
|
|
16712
|
-
|
|
16713
|
-
whiteModelLineGroup = new THREE.Group();
|
|
16714
|
-
// 原始数据白模
|
|
16715
|
-
originalWhiteMode = new THREE.Group();
|
|
16716
|
-
material = new THREE.MeshStandardMaterial({ color: 16777215, transparent: true, opacity: 0.8, side: THREE.DoubleSide });
|
|
16717
|
-
itemList = [];
|
|
16718
|
-
promise;
|
|
16719
|
-
/** 设置物品列表
|
|
16720
|
-
* @param itemList
|
|
16721
|
-
*/
|
|
16722
|
-
setItemList(itemList) {
|
|
16723
|
-
if (Array.isArray(itemList)) this.itemList = itemList;
|
|
16724
|
-
else if ("itemInfo" in itemList) this.itemList = itemList.itemInfo;
|
|
16725
|
-
}
|
|
16107
|
+
desPoints = [];
|
|
16108
|
+
raylines = [];
|
|
16109
|
+
data = [];
|
|
16726
16110
|
onAddFromParent(parent) {
|
|
16727
16111
|
this.Dxf = parent.findComponentByName("Dxf");
|
|
16728
16112
|
this.Variable = parent.findComponentByName("Variable");
|
|
16729
|
-
this.
|
|
16730
|
-
|
|
16731
|
-
this.promise = this.updateModel();
|
|
16113
|
+
this.Dxf?.addEventListener("setDta", () => {
|
|
16114
|
+
this.updateModel();
|
|
16732
16115
|
});
|
|
16733
16116
|
}
|
|
16734
|
-
|
|
16735
|
-
|
|
16736
|
-
|
|
16737
|
-
|
|
16738
|
-
|
|
16739
|
-
|
|
16740
|
-
|
|
16741
|
-
|
|
16742
|
-
|
|
16743
|
-
|
|
16744
|
-
|
|
16745
|
-
|
|
16746
|
-
|
|
16747
|
-
|
|
16748
|
-
|
|
16749
|
-
|
|
16117
|
+
/**
|
|
16118
|
+
* 设置值
|
|
16119
|
+
* @param data
|
|
16120
|
+
*/
|
|
16121
|
+
async set(data) {
|
|
16122
|
+
if (typeof data === "string") {
|
|
16123
|
+
if (typeof global !== "undefined") {
|
|
16124
|
+
const packageName = "fs";
|
|
16125
|
+
const { default: fs2 } = await import(
|
|
16126
|
+
/* @vite-ignore */
|
|
16127
|
+
packageName
|
|
16128
|
+
);
|
|
16129
|
+
const buffer = fs2.readFileSync(data);
|
|
16130
|
+
const json = JSON.parse(buffer.toString("utf-8"));
|
|
16131
|
+
this.set(json);
|
|
16132
|
+
return;
|
|
16133
|
+
} else {
|
|
16134
|
+
throw new Error("非node环境不允许使用路径");
|
|
16135
|
+
}
|
|
16750
16136
|
}
|
|
16751
|
-
|
|
16752
|
-
|
|
16753
|
-
|
|
16754
|
-
|
|
16755
|
-
|
|
16756
|
-
|
|
16757
|
-
|
|
16758
|
-
|
|
16759
|
-
|
|
16760
|
-
|
|
16761
|
-
|
|
16762
|
-
|
|
16763
|
-
|
|
16764
|
-
|
|
16765
|
-
|
|
16137
|
+
this.data = data;
|
|
16138
|
+
this.updateModel();
|
|
16139
|
+
}
|
|
16140
|
+
/**
|
|
16141
|
+
* 设置射线辅助
|
|
16142
|
+
*/
|
|
16143
|
+
racasterHelper(position, direction, far) {
|
|
16144
|
+
this.raylines.push([
|
|
16145
|
+
position.clone(),
|
|
16146
|
+
position.clone().add(direction.clone().multiplyScalar(far))
|
|
16147
|
+
]);
|
|
16148
|
+
direction.z = 0;
|
|
16149
|
+
this.raylines.push([
|
|
16150
|
+
position.clone(),
|
|
16151
|
+
position.clone().add(direction.clone().multiplyScalar(far))
|
|
16152
|
+
]);
|
|
16153
|
+
}
|
|
16154
|
+
_timer = null;
|
|
16155
|
+
/**
|
|
16156
|
+
* 更新模型
|
|
16157
|
+
*/
|
|
16158
|
+
updateModel() {
|
|
16159
|
+
if (this._timer) clearTimeout(this._timer);
|
|
16160
|
+
this._timer = setTimeout(() => {
|
|
16161
|
+
this._timer = null;
|
|
16162
|
+
const whiteModel = this.parent?.findComponentByName("WhiteModel");
|
|
16163
|
+
this.raylines.length = 0;
|
|
16164
|
+
this.desPoints.length = 0;
|
|
16165
|
+
this.data.forEach((item) => {
|
|
16166
|
+
const position = new THREE.Vector3(
|
|
16167
|
+
item.position.x,
|
|
16168
|
+
item.position.y,
|
|
16169
|
+
item.position.z
|
|
16170
|
+
);
|
|
16171
|
+
const direction = new THREE.Vector3(
|
|
16172
|
+
item.direction.x,
|
|
16173
|
+
item.direction.y,
|
|
16174
|
+
item.direction.z
|
|
16175
|
+
);
|
|
16176
|
+
const far = 100;
|
|
16177
|
+
this.racasterHelper(position, direction, far);
|
|
16178
|
+
direction.z = 0;
|
|
16179
|
+
const raycaster = new THREE.Raycaster(position, direction, 0, far);
|
|
16180
|
+
const list = raycaster.intersectObject(whiteModel.originalWhiteMode);
|
|
16181
|
+
if (list.length) {
|
|
16182
|
+
const { point } = list[0];
|
|
16183
|
+
this.desPoints.push({
|
|
16184
|
+
message: item.desc,
|
|
16185
|
+
position,
|
|
16186
|
+
intersection: point
|
|
16187
|
+
});
|
|
16188
|
+
}
|
|
16766
16189
|
});
|
|
16767
|
-
|
|
16768
|
-
|
|
16769
|
-
|
|
16190
|
+
this.dispatchEvent({
|
|
16191
|
+
type: "handleSuccess",
|
|
16192
|
+
desPoints: this.desPoints
|
|
16193
|
+
});
|
|
16194
|
+
}, 50);
|
|
16195
|
+
}
|
|
16196
|
+
}
|
|
16197
|
+
class DxfLineModel extends Component {
|
|
16198
|
+
static name = "DxfLineModel";
|
|
16199
|
+
dxfLineModel = new THREE.LineSegments();
|
|
16200
|
+
dxfDoorsLineModel = new THREE.LineSegments();
|
|
16201
|
+
dxfModelGroup = new THREE.Group();
|
|
16202
|
+
onAddFromParent(parent) {
|
|
16203
|
+
const dxf = parent.findComponentByName("Dxf");
|
|
16204
|
+
this.dxfModelGroup.add(this.dxfLineModel);
|
|
16205
|
+
this.dxfModelGroup.add(this.dxfDoorsLineModel);
|
|
16206
|
+
this.dxfDoorsLineModel.material = new THREE.LineBasicMaterial({ color: 16776960, vertexColors: true });
|
|
16207
|
+
dxf?.addEventListener("lineOffset", () => this.updateMode());
|
|
16208
|
+
}
|
|
16209
|
+
updateMode() {
|
|
16210
|
+
const dxf = this.parent?.findComponentByName("Dxf");
|
|
16211
|
+
this.dxfLineModel.clear();
|
|
16212
|
+
const dxfArray = dxf.to3DArray(1 / dxf.scale, 0);
|
|
16213
|
+
this.dxfLineModel.geometry = new THREE.BufferGeometry().setAttribute("position", new THREE.BufferAttribute(dxfArray, 3, true));
|
|
16214
|
+
const doorsArray = new Float32Array(
|
|
16215
|
+
dxf.doorLineSegment.flatMap(({ start, end }) => [start.x, start.y, 0, end.x, end.y, 0])
|
|
16216
|
+
).map((n) => n / dxf.scale);
|
|
16217
|
+
const doorsColorArray = new Float32Array(dxf.doorLineSegment.flatMap(() => [1, 0, 0, 0, 1, 0]));
|
|
16218
|
+
this.dxfDoorsLineModel.geometry = new THREE.BufferGeometry().setAttribute("position", new THREE.BufferAttribute(doorsArray, 3, true)).setAttribute("color", new THREE.BufferAttribute(doorsColorArray, 3));
|
|
16219
|
+
this.dxfModelGroup.position.z = dxf.originalZAverage;
|
|
16220
|
+
this.dispatchEvent({
|
|
16221
|
+
type: "modelUpdate",
|
|
16222
|
+
model: this.dxfModelGroup
|
|
16223
|
+
});
|
|
16224
|
+
}
|
|
16225
|
+
}
|
|
16226
|
+
const index$2 = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
|
|
16227
|
+
__proto__: null,
|
|
16228
|
+
DetailsPoint,
|
|
16229
|
+
DxfLineModel,
|
|
16230
|
+
WhiteModel
|
|
16231
|
+
}, Symbol.toStringTag, { value: "Module" }));
|
|
16232
|
+
function ModelDataPlugin_(dxfSystem, option = {}) {
|
|
16233
|
+
const {
|
|
16234
|
+
detailsPoint = true,
|
|
16235
|
+
whiteModel = true,
|
|
16236
|
+
dxfLineModel = true
|
|
16237
|
+
} = option;
|
|
16238
|
+
dxfLineModel && dxfSystem.addComponent(new DxfLineModel());
|
|
16239
|
+
whiteModel && dxfSystem.addComponent(new WhiteModel());
|
|
16240
|
+
detailsPoint && dxfSystem.addComponent(new DetailsPoint());
|
|
16241
|
+
}
|
|
16242
|
+
const ModelDataPlugin = Object.assign(ModelDataPlugin_, {
|
|
16243
|
+
create(option = {}) {
|
|
16244
|
+
return (dxfSystem) => ModelDataPlugin_(dxfSystem, option);
|
|
16245
|
+
}
|
|
16246
|
+
});
|
|
16247
|
+
const index$1 = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
|
|
16248
|
+
__proto__: null,
|
|
16249
|
+
ModelDataPlugin,
|
|
16250
|
+
components: index$2
|
|
16251
|
+
}, Symbol.toStringTag, { value: "Module" }));
|
|
16252
|
+
class Lines extends THREE.LineSegments {
|
|
16253
|
+
geometry = new THREE.BufferGeometry();
|
|
16254
|
+
points = [];
|
|
16255
|
+
pointsObject3D;
|
|
16256
|
+
constructor(points = [], color = 16777215) {
|
|
16257
|
+
super();
|
|
16258
|
+
this.geometry = this.geometry;
|
|
16259
|
+
this.addPoint(...points);
|
|
16260
|
+
this.frustumCulled = false;
|
|
16261
|
+
this.pointsObject3D = new THREE.Points(this.geometry, new THREE.PointsMaterial({
|
|
16262
|
+
sizeAttenuation: false,
|
|
16263
|
+
size: 10
|
|
16264
|
+
}));
|
|
16265
|
+
this.material = new THREE.LineBasicMaterial({ color });
|
|
16266
|
+
}
|
|
16267
|
+
addPoint(...points) {
|
|
16268
|
+
this.points.push(...points);
|
|
16269
|
+
this.updateGeometry();
|
|
16270
|
+
}
|
|
16271
|
+
setPoint(...points) {
|
|
16272
|
+
this.points.length = 0;
|
|
16273
|
+
this.addPoint(...points);
|
|
16274
|
+
}
|
|
16275
|
+
_timer = null;
|
|
16276
|
+
updateGeometry() {
|
|
16277
|
+
if (this._timer) clearTimeout(this._timer);
|
|
16278
|
+
this._timer = setTimeout(() => {
|
|
16279
|
+
const array = this.points.flatMap((p, i) => {
|
|
16280
|
+
if (i === 0) return [];
|
|
16281
|
+
else {
|
|
16282
|
+
const p0 = this.points[i - 1];
|
|
16283
|
+
return [p0.x, p0.y, p0.z, p.x, p.y, p.z];
|
|
16284
|
+
}
|
|
16770
16285
|
});
|
|
16771
|
-
|
|
16772
|
-
|
|
16773
|
-
this.
|
|
16774
|
-
});
|
|
16775
|
-
this.dispatchEvent({
|
|
16776
|
-
type: "updateModel",
|
|
16777
|
-
originalWhiteMode: this.originalWhiteMode,
|
|
16778
|
-
whiteModelGroup: this.whiteModelGroup
|
|
16286
|
+
const position = new THREE.BufferAttribute(new Float32Array(array), 3);
|
|
16287
|
+
this.geometry.setAttribute("position", position);
|
|
16288
|
+
this._timer = null;
|
|
16779
16289
|
});
|
|
16780
|
-
return this.whiteModelLineGroup;
|
|
16781
16290
|
}
|
|
16782
|
-
|
|
16783
|
-
|
|
16784
|
-
|
|
16785
|
-
|
|
16786
|
-
|
|
16291
|
+
}
|
|
16292
|
+
function selectLocalFileFun() {
|
|
16293
|
+
return new Promise((resolve) => {
|
|
16294
|
+
const input = document.createElement("input");
|
|
16295
|
+
input.type = "file";
|
|
16296
|
+
input.accept = "application/json";
|
|
16297
|
+
input.click();
|
|
16298
|
+
input.onchange = () => {
|
|
16299
|
+
if (input.files?.length) resolve(input.files[0]);
|
|
16300
|
+
else resolve(null);
|
|
16301
|
+
};
|
|
16302
|
+
});
|
|
16303
|
+
}
|
|
16304
|
+
const SelectLocalFile = Object.assign(selectLocalFileFun, {
|
|
16305
|
+
arrayBuffer() {
|
|
16787
16306
|
return new Promise(async (resolve) => {
|
|
16788
|
-
await
|
|
16789
|
-
|
|
16790
|
-
|
|
16791
|
-
|
|
16792
|
-
|
|
16793
|
-
|
|
16794
|
-
|
|
16795
|
-
}
|
|
16307
|
+
const file = await selectLocalFileFun();
|
|
16308
|
+
if (file instanceof File) {
|
|
16309
|
+
const fileReader = new FileReader();
|
|
16310
|
+
fileReader.onload = () => {
|
|
16311
|
+
resolve(fileReader.result);
|
|
16312
|
+
};
|
|
16313
|
+
fileReader.readAsArrayBuffer(file);
|
|
16314
|
+
} else resolve(null);
|
|
16796
16315
|
});
|
|
16797
|
-
}
|
|
16798
|
-
|
|
16799
|
-
* 转为 glb
|
|
16800
|
-
* @param binary
|
|
16801
|
-
* @returns
|
|
16802
|
-
*/
|
|
16803
|
-
toGltf(binary = true) {
|
|
16804
|
-
if (!glbExporter) glbExporter = new gltf.GLTFExporter();
|
|
16316
|
+
},
|
|
16317
|
+
text() {
|
|
16805
16318
|
return new Promise(async (resolve) => {
|
|
16806
|
-
await
|
|
16807
|
-
|
|
16808
|
-
|
|
16809
|
-
|
|
16810
|
-
|
|
16811
|
-
|
|
16812
|
-
|
|
16813
|
-
|
|
16814
|
-
}, () => {
|
|
16815
|
-
resolve(void 0);
|
|
16816
|
-
}, {
|
|
16817
|
-
binary
|
|
16818
|
-
});
|
|
16819
|
-
}, 20);
|
|
16319
|
+
const file = await selectLocalFileFun();
|
|
16320
|
+
if (file instanceof File) {
|
|
16321
|
+
const fileReader = new FileReader();
|
|
16322
|
+
fileReader.onload = () => {
|
|
16323
|
+
resolve(fileReader.result);
|
|
16324
|
+
};
|
|
16325
|
+
fileReader.readAsText(file, "utf-8");
|
|
16326
|
+
} else resolve(null);
|
|
16820
16327
|
});
|
|
16328
|
+
},
|
|
16329
|
+
async json() {
|
|
16330
|
+
const text = await this.text();
|
|
16331
|
+
if (text) return JSON.parse(text);
|
|
16332
|
+
}
|
|
16333
|
+
});
|
|
16334
|
+
class CommandFlow extends EventDispatcher {
|
|
16335
|
+
list = [];
|
|
16336
|
+
rollbacklist = [];
|
|
16337
|
+
revokeRollbacklist = [];
|
|
16338
|
+
// 是否写入操作记录
|
|
16339
|
+
writeOperationList = true;
|
|
16340
|
+
loop = false;
|
|
16341
|
+
setLoop(loop) {
|
|
16342
|
+
this.loop = loop;
|
|
16343
|
+
return this;
|
|
16821
16344
|
}
|
|
16822
16345
|
/**
|
|
16823
|
-
*
|
|
16346
|
+
*
|
|
16347
|
+
* @param operation
|
|
16824
16348
|
* @returns
|
|
16825
16349
|
*/
|
|
16826
|
-
|
|
16827
|
-
|
|
16828
|
-
|
|
16829
|
-
return new Blob([buffer], { type: "application/octet-stream" });
|
|
16830
|
-
}
|
|
16350
|
+
add(operation) {
|
|
16351
|
+
this.list.push(operation);
|
|
16352
|
+
return this;
|
|
16831
16353
|
}
|
|
16832
|
-
/**
|
|
16833
|
-
*
|
|
16354
|
+
/** 添加回滚回调列表
|
|
16355
|
+
* @param callBack
|
|
16356
|
+
*/
|
|
16357
|
+
addRollback(callBack) {
|
|
16358
|
+
this.rollbacklist.push(callBack);
|
|
16359
|
+
return this;
|
|
16360
|
+
}
|
|
16361
|
+
/** 添加撤回回滚回调列表
|
|
16362
|
+
* @param callBack
|
|
16834
16363
|
* @returns
|
|
16835
16364
|
*/
|
|
16836
|
-
|
|
16837
|
-
|
|
16838
|
-
|
|
16839
|
-
|
|
16840
|
-
|
|
16841
|
-
|
|
16365
|
+
addRevokeRollback(callBack) {
|
|
16366
|
+
this.revokeRollbacklist.push(callBack);
|
|
16367
|
+
return this;
|
|
16368
|
+
}
|
|
16369
|
+
}
|
|
16370
|
+
class CommandManager extends EventDispatcher {
|
|
16371
|
+
commandFlowMap = /* @__PURE__ */ new Map();
|
|
16372
|
+
lock = false;
|
|
16373
|
+
abortController = null;
|
|
16374
|
+
resolve = null;
|
|
16375
|
+
currentName = null;
|
|
16376
|
+
_disabled = false;
|
|
16377
|
+
set disabled(disabled) {
|
|
16378
|
+
this._disabled = disabled;
|
|
16379
|
+
if (this._disabled) this.cancel();
|
|
16380
|
+
}
|
|
16381
|
+
get disabled() {
|
|
16382
|
+
return this._disabled;
|
|
16842
16383
|
}
|
|
16843
16384
|
/**
|
|
16844
|
-
*
|
|
16845
|
-
|
|
16385
|
+
* 操作记录
|
|
16386
|
+
*/
|
|
16387
|
+
operationList = [];
|
|
16388
|
+
rollbackList = [];
|
|
16389
|
+
constructor() {
|
|
16390
|
+
super();
|
|
16391
|
+
}
|
|
16392
|
+
/** 添加命令流
|
|
16393
|
+
* @param name
|
|
16846
16394
|
* @returns
|
|
16847
16395
|
*/
|
|
16848
|
-
|
|
16849
|
-
if (
|
|
16850
|
-
|
|
16851
|
-
|
|
16852
|
-
|
|
16853
|
-
a.href = URL.createObjectURL(blob);
|
|
16854
|
-
a.download = filename;
|
|
16855
|
-
a.click();
|
|
16856
|
-
} else if (typeof global !== "undefined") {
|
|
16857
|
-
const buffer = await this.toOBJ();
|
|
16858
|
-
if (buffer) {
|
|
16859
|
-
const fs2 = await include("fs", false);
|
|
16860
|
-
fs2.writeFileSync(filename, buffer);
|
|
16861
|
-
}
|
|
16862
|
-
}
|
|
16396
|
+
addCommandFlow(name) {
|
|
16397
|
+
if (this.commandFlowMap.has(name)) throw new Error(`${name} 命令已经存在`);
|
|
16398
|
+
const commandFlow = new CommandFlow();
|
|
16399
|
+
this.commandFlowMap.set(name, commandFlow);
|
|
16400
|
+
return commandFlow;
|
|
16863
16401
|
}
|
|
16864
|
-
|
|
16865
|
-
|
|
16866
|
-
|
|
16402
|
+
executionPromise = null;
|
|
16403
|
+
executionResolve = null;
|
|
16404
|
+
/** 执行控制流
|
|
16405
|
+
* @param name
|
|
16867
16406
|
* @returns
|
|
16868
16407
|
*/
|
|
16869
|
-
async
|
|
16870
|
-
if (
|
|
16871
|
-
|
|
16872
|
-
|
|
16873
|
-
|
|
16874
|
-
|
|
16875
|
-
|
|
16876
|
-
|
|
16877
|
-
|
|
16878
|
-
|
|
16879
|
-
|
|
16880
|
-
|
|
16881
|
-
|
|
16408
|
+
async start(name, data = null, step = 0) {
|
|
16409
|
+
if (this.disabled) throw new Error("命令管理器已禁用,无法启动新的命令流");
|
|
16410
|
+
this.dispatchEvent({
|
|
16411
|
+
type: "startedBefore",
|
|
16412
|
+
name,
|
|
16413
|
+
currentName: this.currentName
|
|
16414
|
+
});
|
|
16415
|
+
this.executionPromise && await this.executionPromise;
|
|
16416
|
+
this.executionPromise = null;
|
|
16417
|
+
if (this.lock) {
|
|
16418
|
+
throw new Error("命令管理器已被 " + this.currentName + " 命令锁定,无法启动新的命令流,请退出或等待命令执行结束");
|
|
16419
|
+
}
|
|
16420
|
+
const commandFlow = this.commandFlowMap.get(name);
|
|
16421
|
+
if (!commandFlow) {
|
|
16422
|
+
throw new Error(`命令流 ${name} 不存在`);
|
|
16423
|
+
}
|
|
16424
|
+
this.lock = true;
|
|
16425
|
+
this.abortController = new AbortController();
|
|
16426
|
+
this.currentName = name;
|
|
16427
|
+
commandFlow.dispatchEvent({ type: "started" });
|
|
16428
|
+
this.dispatchEvent({ type: "started", name });
|
|
16429
|
+
try {
|
|
16430
|
+
for (let i = step; i < commandFlow.list.length; i++) {
|
|
16431
|
+
const operation = commandFlow.list[i];
|
|
16432
|
+
commandFlow.dispatchEvent({ type: "executing", index: i });
|
|
16433
|
+
this.dispatchEvent({ type: "executing", name, index: i });
|
|
16434
|
+
data = await new Promise((resolve) => {
|
|
16435
|
+
this.resolve = resolve;
|
|
16436
|
+
operation(resolve, data);
|
|
16437
|
+
});
|
|
16438
|
+
if (this.abortController.signal.aborted) {
|
|
16439
|
+
commandFlow.dispatchEvent({ type: "executionInterrupt", index: i });
|
|
16440
|
+
this.dispatchEvent({ type: "executionInterrupt", name, index: i });
|
|
16441
|
+
this.dispatchEvent({ type: "cancel", name });
|
|
16442
|
+
break;
|
|
16443
|
+
} else {
|
|
16444
|
+
commandFlow.dispatchEvent({ type: "executionCompleted", index: i, data });
|
|
16445
|
+
this.dispatchEvent({ type: "executionCompleted", name, index: i, data });
|
|
16446
|
+
}
|
|
16447
|
+
}
|
|
16448
|
+
} catch (error) {
|
|
16449
|
+
console.error(error);
|
|
16450
|
+
} finally {
|
|
16451
|
+
this.lock = false;
|
|
16452
|
+
this.currentName = null;
|
|
16453
|
+
if (this.abortController && !this.abortController.signal.aborted) {
|
|
16454
|
+
commandFlow.dispatchEvent({ type: "completed", data });
|
|
16455
|
+
this.dispatchEvent({ type: "completed", name, data });
|
|
16456
|
+
if (commandFlow.writeOperationList) {
|
|
16457
|
+
this.operationList.push({ name, data });
|
|
16458
|
+
this.rollbackList.length = 0;
|
|
16459
|
+
}
|
|
16460
|
+
if (commandFlow.loop) queueMicrotask(() => this.start(name));
|
|
16461
|
+
}
|
|
16462
|
+
this.abortController = null;
|
|
16463
|
+
commandFlow.dispatchEvent({ type: "finally" });
|
|
16464
|
+
this.dispatchEvent({ type: "finally", name });
|
|
16465
|
+
if (this.executionResolve) {
|
|
16466
|
+
this.executionResolve(null);
|
|
16467
|
+
this.executionResolve = null;
|
|
16882
16468
|
}
|
|
16883
16469
|
}
|
|
16470
|
+
return data;
|
|
16884
16471
|
}
|
|
16885
|
-
|
|
16886
|
-
class DetailsPoint extends Component {
|
|
16887
|
-
static name = "DetailsPoint";
|
|
16888
|
-
Dxf = null;
|
|
16889
|
-
WhiteModel = null;
|
|
16890
|
-
Variable = null;
|
|
16891
|
-
desPoints = [];
|
|
16892
|
-
raylines = [];
|
|
16893
|
-
data = [];
|
|
16894
|
-
onAddFromParent(parent) {
|
|
16895
|
-
this.Dxf = parent.findComponentByName("Dxf");
|
|
16896
|
-
this.Variable = parent.findComponentByName("Variable");
|
|
16897
|
-
this.Dxf?.addEventListener("setDta", () => {
|
|
16898
|
-
this.updateModel();
|
|
16899
|
-
});
|
|
16900
|
-
}
|
|
16901
|
-
/**
|
|
16902
|
-
* 设置值
|
|
16903
|
-
* @param data
|
|
16472
|
+
/** 取消当前命令
|
|
16904
16473
|
*/
|
|
16905
|
-
|
|
16906
|
-
if (
|
|
16907
|
-
|
|
16908
|
-
|
|
16909
|
-
|
|
16910
|
-
/* @vite-ignore */
|
|
16911
|
-
packageName
|
|
16912
|
-
);
|
|
16913
|
-
const buffer = fs2.readFileSync(data);
|
|
16914
|
-
const json = JSON.parse(buffer.toString("utf-8"));
|
|
16915
|
-
this.set(json);
|
|
16916
|
-
return;
|
|
16917
|
-
} else {
|
|
16918
|
-
throw new Error("非node环境不允许使用路径");
|
|
16919
|
-
}
|
|
16474
|
+
cancel() {
|
|
16475
|
+
if (this.abortController) {
|
|
16476
|
+
this.abortController.abort();
|
|
16477
|
+
if (this.resolve) this.resolve();
|
|
16478
|
+
this.executionPromise = new Promise((resolve) => this.executionResolve = resolve);
|
|
16920
16479
|
}
|
|
16921
|
-
this.data = data;
|
|
16922
|
-
this.updateModel();
|
|
16923
16480
|
}
|
|
16924
16481
|
/**
|
|
16925
|
-
*
|
|
16482
|
+
* 回滚
|
|
16926
16483
|
*/
|
|
16927
|
-
|
|
16928
|
-
|
|
16929
|
-
|
|
16930
|
-
|
|
16931
|
-
|
|
16932
|
-
|
|
16933
|
-
|
|
16934
|
-
|
|
16935
|
-
|
|
16936
|
-
|
|
16484
|
+
rollback() {
|
|
16485
|
+
try {
|
|
16486
|
+
const operation = this.operationList.pop();
|
|
16487
|
+
if (!operation) return false;
|
|
16488
|
+
const commandFlow = this.commandFlowMap.get(operation.name);
|
|
16489
|
+
if (!commandFlow) return false;
|
|
16490
|
+
const data = commandFlow.rollbacklist.reduce((data2, callBack) => callBack(data2), operation.data);
|
|
16491
|
+
this.dispatchEvent({ type: "rollback", name: operation.name });
|
|
16492
|
+
this.rollbackList.push({
|
|
16493
|
+
data,
|
|
16494
|
+
name: operation.name
|
|
16495
|
+
});
|
|
16496
|
+
return true;
|
|
16497
|
+
} catch (error) {
|
|
16498
|
+
throw new Error(`回滚失败:${error}`);
|
|
16499
|
+
}
|
|
16937
16500
|
}
|
|
16938
|
-
_timer = null;
|
|
16939
16501
|
/**
|
|
16940
|
-
*
|
|
16502
|
+
* 撤销回滚
|
|
16941
16503
|
*/
|
|
16942
|
-
|
|
16943
|
-
|
|
16944
|
-
|
|
16945
|
-
|
|
16946
|
-
const
|
|
16947
|
-
|
|
16948
|
-
|
|
16949
|
-
this.
|
|
16950
|
-
|
|
16951
|
-
|
|
16952
|
-
|
|
16953
|
-
|
|
16954
|
-
|
|
16955
|
-
const direction = new THREE.Vector3(
|
|
16956
|
-
item.direction.x,
|
|
16957
|
-
item.direction.y,
|
|
16958
|
-
item.direction.z
|
|
16959
|
-
);
|
|
16960
|
-
const far = 100;
|
|
16961
|
-
this.racasterHelper(position, direction, far);
|
|
16962
|
-
direction.z = 0;
|
|
16963
|
-
const raycaster = new THREE.Raycaster(position, direction, 0, far);
|
|
16964
|
-
const list = raycaster.intersectObject(whiteModel.originalWhiteMode);
|
|
16965
|
-
if (list.length) {
|
|
16966
|
-
const { point } = list[0];
|
|
16967
|
-
this.desPoints.push({
|
|
16968
|
-
message: item.desc,
|
|
16969
|
-
position,
|
|
16970
|
-
intersection: point
|
|
16971
|
-
});
|
|
16972
|
-
}
|
|
16973
|
-
});
|
|
16974
|
-
this.dispatchEvent({
|
|
16975
|
-
type: "handleSuccess",
|
|
16976
|
-
desPoints: this.desPoints
|
|
16977
|
-
});
|
|
16978
|
-
}, 50);
|
|
16979
|
-
}
|
|
16980
|
-
}
|
|
16981
|
-
class DxfLineModel extends Component {
|
|
16982
|
-
static name = "DxfLineModel";
|
|
16983
|
-
dxfLineModel = new THREE.LineSegments();
|
|
16984
|
-
dxfDoorsLineModel = new THREE.LineSegments();
|
|
16985
|
-
dxfModelGroup = new THREE.Group();
|
|
16986
|
-
onAddFromParent(parent) {
|
|
16987
|
-
const dxf = parent.findComponentByName("Dxf");
|
|
16988
|
-
this.dxfModelGroup.add(this.dxfLineModel);
|
|
16989
|
-
this.dxfModelGroup.add(this.dxfDoorsLineModel);
|
|
16990
|
-
this.dxfDoorsLineModel.material = new THREE.LineBasicMaterial({ color: 16776960, vertexColors: true });
|
|
16991
|
-
dxf?.addEventListener("lineOffset", () => this.updateMode());
|
|
16504
|
+
revokeRollback() {
|
|
16505
|
+
try {
|
|
16506
|
+
const operation = this.rollbackList.pop();
|
|
16507
|
+
if (!operation) return false;
|
|
16508
|
+
const commandFlow = this.commandFlowMap.get(operation.name);
|
|
16509
|
+
if (!commandFlow) return false;
|
|
16510
|
+
const data = commandFlow.revokeRollbacklist.reduce((data2, callBack) => callBack(data2), operation.data);
|
|
16511
|
+
this.dispatchEvent({ type: "revokeRollback", name: operation.name });
|
|
16512
|
+
this.operationList.push({ name: operation.name, data });
|
|
16513
|
+
return true;
|
|
16514
|
+
} catch (error) {
|
|
16515
|
+
throw new Error(`撤回回滚失败:${error}`);
|
|
16516
|
+
}
|
|
16992
16517
|
}
|
|
16993
|
-
|
|
16994
|
-
|
|
16995
|
-
this.
|
|
16996
|
-
|
|
16997
|
-
|
|
16998
|
-
|
|
16999
|
-
|
|
17000
|
-
|
|
17001
|
-
|
|
17002
|
-
|
|
17003
|
-
|
|
17004
|
-
|
|
17005
|
-
|
|
17006
|
-
model: this.dxfModelGroup
|
|
17007
|
-
});
|
|
16518
|
+
destroy() {
|
|
16519
|
+
this.cancel();
|
|
16520
|
+
this.addEventListener("cancel", () => {
|
|
16521
|
+
this.commandFlowMap.clear();
|
|
16522
|
+
this.lock = false;
|
|
16523
|
+
this.abortController = null;
|
|
16524
|
+
this.resolve = null;
|
|
16525
|
+
this.executionResolve = null;
|
|
16526
|
+
this.executionPromise = null;
|
|
16527
|
+
this.currentName = null;
|
|
16528
|
+
this.operationList.length = 0;
|
|
16529
|
+
this.rollbackList.length = 0;
|
|
16530
|
+
}, { once: true });
|
|
17008
16531
|
}
|
|
17009
16532
|
}
|
|
17010
|
-
const index$1 = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
|
|
17011
|
-
__proto__: null,
|
|
17012
|
-
DetailsPoint,
|
|
17013
|
-
DxfLineModel,
|
|
17014
|
-
WhiteModel
|
|
17015
|
-
}, Symbol.toStringTag, { value: "Module" }));
|
|
17016
|
-
function ModelDataPlugin_(dxfSystem, option = {}) {
|
|
17017
|
-
const {
|
|
17018
|
-
detailsPoint = true,
|
|
17019
|
-
whiteModel = true,
|
|
17020
|
-
dxfLineModel = true
|
|
17021
|
-
} = option;
|
|
17022
|
-
dxfLineModel && dxfSystem.addComponent(new DxfLineModel());
|
|
17023
|
-
whiteModel && dxfSystem.addComponent(new WhiteModel());
|
|
17024
|
-
detailsPoint && dxfSystem.addComponent(new DetailsPoint());
|
|
17025
|
-
}
|
|
17026
|
-
const ModelDataPlugin = Object.assign(ModelDataPlugin_, {
|
|
17027
|
-
create(option = {}) {
|
|
17028
|
-
return (dxfSystem) => ModelDataPlugin_(dxfSystem, option);
|
|
17029
|
-
}
|
|
17030
|
-
});
|
|
17031
16533
|
const index = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
|
|
17032
16534
|
__proto__: null,
|
|
17033
|
-
|
|
17034
|
-
|
|
16535
|
+
ArrayMap,
|
|
16536
|
+
AxisAlignCorr,
|
|
16537
|
+
BoundExt,
|
|
16538
|
+
Box2,
|
|
16539
|
+
CommandFlow,
|
|
16540
|
+
CommandManager,
|
|
16541
|
+
Component,
|
|
16542
|
+
ComponentManager,
|
|
16543
|
+
CountMap,
|
|
16544
|
+
DoubleWallHelper,
|
|
16545
|
+
DxfSystem,
|
|
16546
|
+
EventDispatcher,
|
|
16547
|
+
GltfInit,
|
|
16548
|
+
LineGroupType,
|
|
16549
|
+
LineIndexGenerator,
|
|
16550
|
+
LineSegment,
|
|
16551
|
+
LineSegmentUndirectedGraph,
|
|
16552
|
+
Lines,
|
|
16553
|
+
MapEnhance,
|
|
16554
|
+
Point,
|
|
16555
|
+
PointVirtualGrid,
|
|
16556
|
+
Polygon,
|
|
16557
|
+
PvgList,
|
|
16558
|
+
Quadtree,
|
|
16559
|
+
Rectangle,
|
|
16560
|
+
SceneAutoGenerat,
|
|
16561
|
+
SelectLocalFile,
|
|
16562
|
+
SetMap,
|
|
16563
|
+
UndirectedGraph,
|
|
16564
|
+
UnionFindSet,
|
|
16565
|
+
buildBayWindowGroup,
|
|
16566
|
+
buildDoubleWallGroup,
|
|
16567
|
+
clippingDoubleWall,
|
|
16568
|
+
clippingLineUserData,
|
|
16569
|
+
cloneUserData,
|
|
16570
|
+
closedPathArea,
|
|
16571
|
+
components,
|
|
16572
|
+
findCircleEdges,
|
|
16573
|
+
findClosedPolygons,
|
|
16574
|
+
findDiscretePoint,
|
|
16575
|
+
findDiscretePointLine,
|
|
16576
|
+
findDiscretePointLine2,
|
|
16577
|
+
findLargestCircle,
|
|
16578
|
+
findRingEdges,
|
|
16579
|
+
findVerticalReference,
|
|
16580
|
+
gltf,
|
|
16581
|
+
lineDataToOriginalData,
|
|
16582
|
+
lineDataToThreeVJiaJson,
|
|
16583
|
+
lineSegmentClipping,
|
|
16584
|
+
mergeLineUserData,
|
|
16585
|
+
mergeSmallestCircle,
|
|
16586
|
+
mergeWindow,
|
|
16587
|
+
originalDataToLineData,
|
|
16588
|
+
recomputedWindow,
|
|
16589
|
+
recomputedWindowCenter,
|
|
16590
|
+
ringsDeduplication: ringsDeduplication$1,
|
|
16591
|
+
smallestCircle,
|
|
16592
|
+
toOriginalDataItem,
|
|
16593
|
+
tools,
|
|
16594
|
+
uuid
|
|
17035
16595
|
}, Symbol.toStringTag, { value: "Module" }));
|
|
17036
16596
|
function loadRenderPlugin() {
|
|
17037
16597
|
return import("./index2.js");
|
|
@@ -17074,7 +16634,7 @@ if (typeof globalThis !== "undefined") {
|
|
|
17074
16634
|
globalThis.Log = Log;
|
|
17075
16635
|
}
|
|
17076
16636
|
async function createEditor(dom, camera, orbitControls = false, viewPermission) {
|
|
17077
|
-
const mp = await Promise.resolve().then(() => index);
|
|
16637
|
+
const mp = await Promise.resolve().then(() => index$1);
|
|
17078
16638
|
const rp = await loadRenderPlugin();
|
|
17079
16639
|
const editor = await loadEditorPlugin();
|
|
17080
16640
|
const dxfSystem = new DxfSystem().usePlugin(mp.ModelDataPlugin.create({
|
|
@@ -17121,7 +16681,7 @@ async function getModels(originData, trajectory, itemList) {
|
|
|
17121
16681
|
};
|
|
17122
16682
|
}
|
|
17123
16683
|
async function buildJson(opt) {
|
|
17124
|
-
const mp = await Promise.resolve().then(() => index);
|
|
16684
|
+
const mp = await Promise.resolve().then(() => index$1);
|
|
17125
16685
|
const dxfSystem = new DxfSystem();
|
|
17126
16686
|
let { json, trajectory, axisAlignCorrOption, download, itemInfo } = opt;
|
|
17127
16687
|
dxfSystem.usePlugin(mp.ModelDataPlugin);
|
|
@@ -17224,8 +16784,8 @@ export {
|
|
|
17224
16784
|
getFileAll as j,
|
|
17225
16785
|
getGlobalDxfSystem as k,
|
|
17226
16786
|
Dxf as l,
|
|
17227
|
-
index as m,
|
|
17228
|
-
index
|
|
16787
|
+
index$1 as m,
|
|
16788
|
+
index as n,
|
|
17229
16789
|
components as o,
|
|
17230
16790
|
recomputedWindow as r,
|
|
17231
16791
|
uuid as u
|