fix: meshline triangulation bug

This commit is contained in:
izzy.cz 2019-08-14 12:00:10 +08:00
parent 542bc696be
commit 9d530cb27b
2 changed files with 45 additions and 18 deletions

View File

@ -57,12 +57,12 @@
"geometry": { "geometry": {
"type": "LineString", "type": "LineString",
"coordinates": [ "coordinates": [
[120.294931, 30.224459], [120.294931, 30.224459],[120.294931, 30.224459],
[120.194847, 30.224384], [120.194847, 30.224384],[120.194847, 30.224384],[120.194847, 30.224384],
[120.294847, 30.274384], [120.294847, 30.274384],
[120.344847, 30.34384], [120.344847, 30.34384],
[120.364847, 30.374384], [120.364847, 30.374384],
[120.364847, 30.074384], [120.364847, 30.074384],[120.364847, 30.074384],[120.364847, 30.074384],
] ]
} }
}, { }, {
@ -73,7 +73,9 @@
"coordinates": [ "coordinates": [
[120.394931, 30.324459], [120.394931, 30.324459],
[120.494847, 30.324384], [120.494847, 30.324384],
[120.494847, 30.374384], [120.494847, 30.324384],[120.494847, 30.324384],
[120.45, 30.324384],
[120.45, 30.374384],
] ]
} }
}] }]
@ -92,7 +94,19 @@
patternSpacing: 20 patternSpacing: 20
}) })
.render(); .render();
$.get('https://gw.alipayobjects.com/os/basement_prod/3ed18d7c-bce4-48ca-8716-5248b584481d.json', data => {
const linelayer = scene.LineLayer({
zIndex: 2
}).shape('line')
.size([10, 0])
.source(data)
.color('#fff')
.pattern('right')
.style({
patternSpacing: 20
})
.render();
})
}) })
</script> </script>
</body> </body>

View File

@ -26,12 +26,15 @@ function lineSegmentDistance(end, start) {
return Math.sqrt(dx * dx + dy * dy + dz * dz); return Math.sqrt(dx * dx + dy * dy + dz * dz);
} }
function isPointEqual(a, b) {
return a[0] === b[0] && a[1] === b[1];
}
export default function(points, closed, indexOffset) { export default function(points, closed, indexOffset) {
const lineA = [ 0, 0 ]; const lineA = [ 0, 0 ];
const lineB = [ 0, 0 ]; const lineB = [ 0, 0 ];
const tangent = [ 0, 0 ]; const tangent = [ 0, 0 ];
const miter = [ 0, 0 ]; const miter = [ 0, 0 ];
let _lastFlip = -1;
let _started = false; let _started = false;
let _normal = null; let _normal = null;
const tmp = create(); const tmp = create();
@ -54,8 +57,20 @@ export default function(points, closed, indexOffset) {
const index = count; const index = count;
const last = points[i - 1]; const last = points[i - 1];
const cur = points[i]; const cur = points[i];
const next = i < points.length - 1 ? points[i + 1] : null; let next = i < points.length - 1 ? points[i + 1] : null;
const d = lineSegmentDistance(cur, last) + attrDistance[attrDistance.length - 1]; // 如果当前点和前一点相同,跳过
if (isPointEqual(last, cur)) {
continue;
}
if (next) {
let nextIndex = i + 1;
// 找到不相同的下一点
while (next && isPointEqual(cur, next)) {
next = nextIndex < points.length - 1 ? points[++nextIndex] : null;
}
}
const lineDistance = lineSegmentDistance(cur, last);
const d = lineDistance + attrDistance[attrDistance.length - 1];
direction(lineA, cur, last); direction(lineA, cur, last);
@ -94,18 +109,17 @@ export default function(points, closed, indexOffset) {
// 理论上这种情况下 miterLen = Infinity本应通过 isFinite(miterLen) 判断, // 理论上这种情况下 miterLen = Infinity本应通过 isFinite(miterLen) 判断,
// 但是 AMap 投影变换后丢失精度只能通过一个阈值1000判断。 // 但是 AMap 投影变换后丢失精度只能通过一个阈值1000判断。
if (Math.abs(miterLen) > 1000) { if (Math.abs(miterLen) > 1000) {
normal(_normal, lineA);
extrusions(attrPos, out, miters, cur, _normal, 1); extrusions(attrPos, out, miters, cur, _normal, 1);
attrDistance.push(d, d); attrIndex.push(index + 1, index + 2, index + 3);
const indexData = _lastFlip === 1 ? [ index + 1, index + 3, index + 2 ] attrIndex.push(index + 2, index + 4, index + 3);
: [ index, index + 2, index + 3 ]; normal(tmp, lineB);
attrIndex.push(...indexData); copy(_normal, tmp); // store normal for next round
// 避免在 Material 中使用 THREE.DoubleSide extrusions(attrPos, out, miters, cur, _normal, 1);
attrIndex.push(index + 2, index + 3, index + 4); attrDistance.push(d, d, d, d);
count += 2; // the miter is now the normal for our next join
_lastFlip = -1; count += 4;
continue; continue;
} }
@ -147,7 +161,6 @@ export default function(points, closed, indexOffset) {
// the miter is now the normal for our next join // the miter is now the normal for our next join
count += 5; count += 5;
} }
_lastFlip = flip;
} }
} }