mirror of https://gitee.com/antv-l7/antv-l7
fix: meshline triangulation bug
This commit is contained in:
parent
fac18eafd3
commit
30cf3f424a
|
@ -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>
|
||||||
|
|
|
@ -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;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue