mirror of https://gitee.com/antv-l7/antv-l7
Merge branch 'dev-optimize-line' into 'master'
feat: support bevel joint, dashline & anti-alias 主要改进了 3 点: 1. 超过阈值(非常小的锐角)时,miter 接头转成 bevel 接头 2. 支持虚线 demo/dashline.html 3. anti-alias 边缘反走样,可配置模糊半径 See merge request !29
This commit is contained in:
commit
6c0992b6ec
|
@ -30,8 +30,7 @@ const scene = new L7.Scene({
|
||||||
zoom: 14.82,
|
zoom: 14.82,
|
||||||
});
|
});
|
||||||
scene.on('loaded', () => {
|
scene.on('loaded', () => {
|
||||||
$.get('./data/contour.geojson', data => {
|
$.get('https://gw.alipayobjects.com/os/rmsportal/ZVfOvhVCzwBkISNsuKCc.json', data => {
|
||||||
// data.features = data.features.slice(0,1);
|
|
||||||
scene.LineLayer({
|
scene.LineLayer({
|
||||||
zIndex: 2
|
zIndex: 2
|
||||||
})
|
})
|
||||||
|
|
|
@ -0,0 +1,55 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="ie=edge">
|
||||||
|
<meta name="geometry" content="diagram">
|
||||||
|
<link rel="stylesheet" href="./assets/common.css">
|
||||||
|
<title>dashline demo</title>
|
||||||
|
<style>
|
||||||
|
#map { position:absolute; top:0; bottom:0; width:100%; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div id="map"></div>
|
||||||
|
<script src="https://webapi.amap.com/maps?v=1.4.8&key=15cd8a57710d40c9b7c0e3cc120f1200&plugin=Map3D"></script>
|
||||||
|
<script src="./assets/jquery-3.2.1.min.js"></script>
|
||||||
|
<script src="./assets/dat.gui.min.js"></script>
|
||||||
|
<script src="../build/L7.js"></script>
|
||||||
|
<script>
|
||||||
|
|
||||||
|
const color1 = [ 'rgba(37, 140, 249, 0.8)', 'rgba(14, 241, 242, 0.8)', 'rgba(255, 255, 255, 0.8)' ];
|
||||||
|
const scene = new L7.Scene({
|
||||||
|
id: 'map',
|
||||||
|
mapStyle: 'dark', // 样式URL
|
||||||
|
center: [ 102.602992, 23.107329],
|
||||||
|
pitch: 15,
|
||||||
|
zoom: 14.82,
|
||||||
|
});
|
||||||
|
scene.on('loaded', () => {
|
||||||
|
$.get('https://gw.alipayobjects.com/os/rmsportal/ZVfOvhVCzwBkISNsuKCc.json', data => {
|
||||||
|
scene.LineLayer({
|
||||||
|
zIndex: 2
|
||||||
|
})
|
||||||
|
.source(data)
|
||||||
|
.size('ELEV',(value)=>{
|
||||||
|
return [2,(value-1000)*7];
|
||||||
|
})
|
||||||
|
.active(true)
|
||||||
|
.shape('line')
|
||||||
|
.style({
|
||||||
|
lineType: 'dash',
|
||||||
|
dashArray: 200,
|
||||||
|
dashOffset: 0.2,
|
||||||
|
dashRatio: 0.5
|
||||||
|
})
|
||||||
|
.color('ELEV',["#E8FCFF", "#CFF6FF", "#A1E9ff", "#65CEF7", "#3CB1F0", "#2894E0", "#1772c2", "#105CB3", "#0D408C", "#002466"].reverse())
|
||||||
|
.render();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -108,8 +108,9 @@
|
||||||
"earcut": "^2.1.3",
|
"earcut": "^2.1.3",
|
||||||
"fecha": "^2.3.3",
|
"fecha": "^2.3.3",
|
||||||
"gl-matrix": "^2.4.1",
|
"gl-matrix": "^2.4.1",
|
||||||
|
"gl-vec2": "^1.3.0",
|
||||||
"lodash": "^4.17.5",
|
"lodash": "^4.17.5",
|
||||||
"polyline-normals": "^2.0.2",
|
"polyline-miter-util": "^1.0.1",
|
||||||
"rbush": "^2.0.2",
|
"rbush": "^2.0.2",
|
||||||
"simple-statistics": "^7.0.1",
|
"simple-statistics": "^7.0.1",
|
||||||
"supercluster": "^6.0.1",
|
"supercluster": "^6.0.1",
|
||||||
|
|
|
@ -75,7 +75,7 @@ export default class LineBuffer extends BufferBase {
|
||||||
}
|
}
|
||||||
_getMeshLineAttributes() {
|
_getMeshLineAttributes() {
|
||||||
const layerData = this.get('layerData');
|
const layerData = this.get('layerData');
|
||||||
const { lineType } = this.get('style');
|
const { dashArray } = this.get('style');
|
||||||
const positions = [];
|
const positions = [];
|
||||||
const pickingIds = [];
|
const pickingIds = [];
|
||||||
const normal = [];
|
const normal = [];
|
||||||
|
@ -84,10 +84,11 @@ export default class LineBuffer extends BufferBase {
|
||||||
const indexArray = [];
|
const indexArray = [];
|
||||||
const sizes = [];
|
const sizes = [];
|
||||||
const attrDistance = [];
|
const attrDistance = [];
|
||||||
|
const attrDashArray = [];
|
||||||
layerData.forEach(item => {
|
layerData.forEach(item => {
|
||||||
const props = item;
|
const props = item;
|
||||||
const positionCount = positions.length / 3;
|
const positionCount = positions.length / 3;
|
||||||
const attr = lineShape.Line(item.coordinates, props, positionCount, (lineType !== 'soild'));
|
const attr = lineShape.Line(item.coordinates, props, positionCount, dashArray);
|
||||||
positions.push(...attr.positions);
|
positions.push(...attr.positions);
|
||||||
normal.push(...attr.normal);
|
normal.push(...attr.normal);
|
||||||
miter.push(...attr.miter);
|
miter.push(...attr.miter);
|
||||||
|
@ -96,6 +97,7 @@ export default class LineBuffer extends BufferBase {
|
||||||
sizes.push(...attr.sizes);
|
sizes.push(...attr.sizes);
|
||||||
attrDistance.push(...attr.attrDistance);
|
attrDistance.push(...attr.attrDistance);
|
||||||
pickingIds.push(...attr.pickingIds);
|
pickingIds.push(...attr.pickingIds);
|
||||||
|
attrDashArray.push(...attr.dashArray);
|
||||||
});
|
});
|
||||||
return {
|
return {
|
||||||
positions,
|
positions,
|
||||||
|
@ -105,7 +107,8 @@ export default class LineBuffer extends BufferBase {
|
||||||
indexArray,
|
indexArray,
|
||||||
pickingIds,
|
pickingIds,
|
||||||
sizes,
|
sizes,
|
||||||
attrDistance
|
attrDistance,
|
||||||
|
attrDashArray
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,6 +29,12 @@ export default function fillBuffer(layerData) {
|
||||||
throw new Error('Invalid shape type: ' + shape);
|
throw new Error('Invalid shape type: ' + shape);
|
||||||
}
|
}
|
||||||
toPointShapeAttributes(polygon, coordinates, { size, shape, color, id }, attribute);
|
toPointShapeAttributes(polygon, coordinates, { size, shape, color, id }, attribute);
|
||||||
|
// toPointShapeAttributes(polygon, null, {}, attribute);
|
||||||
|
// instanced attributes
|
||||||
|
// attribute.vertices.push(...coordinates);
|
||||||
|
// attribute.a_size.push(...size);
|
||||||
|
// attribute.colors.push(...color);
|
||||||
|
// attribute.pickingIds.push(id);
|
||||||
|
|
||||||
});
|
});
|
||||||
return attribute;
|
return attribute;
|
||||||
|
@ -78,5 +84,8 @@ function toPointShapeAttributes(polygon, geo, style, attribute) {
|
||||||
attribute.colors.push(...color, ...color, ...color);
|
attribute.colors.push(...color, ...color, ...color);
|
||||||
attribute.pickingIds.push(id, id, id);
|
attribute.pickingIds.push(id, id, id);
|
||||||
|
|
||||||
|
// attribute.shapePositions.push(ax, ay, az, bx, by, bz, cx, cy, cz);
|
||||||
|
// attribute.normals.push(nx, ny, nz, nx, ny, nz, nx, ny, nz);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
import * as THREE from '../../core/three';
|
import * as THREE from '../../core/three';
|
||||||
import Material from './material';
|
import Material from './material';
|
||||||
import { getModule } from '../../util/shaderModule';
|
import { getModule, wrapUniforms } from '../../util/shaderModule';
|
||||||
import arcline_frag from '../shader/arcline_frag.glsl';
|
import arcline_frag from '../shader/arcline_frag.glsl';
|
||||||
import arcline_vert from '../shader/arcline_vert.glsl';
|
import arcline_vert from '../shader/arcline_vert.glsl';
|
||||||
|
import merge from '@antv/util/lib/deep-mix';
|
||||||
|
|
||||||
export function LineMaterial(options) {
|
export function LineMaterial(options) {
|
||||||
const { vs, fs } = getModule('line');
|
const { vs, fs } = getModule('line');
|
||||||
|
@ -40,19 +40,14 @@ export function ArcLineMaterial(options) {
|
||||||
return material;
|
return material;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function MeshLineMaterial(options) {
|
export function MeshLineMaterial(options, defines) {
|
||||||
const { vs, fs } = getModule('meshline');
|
const { vs, fs, uniforms } = getModule('meshline');
|
||||||
const material = new Material({
|
const material = new Material({
|
||||||
uniforms: {
|
uniforms: wrapUniforms(merge(uniforms, options, {
|
||||||
u_opacity: { value: options.u_opacity || 1.0 },
|
u_activeId: options.activeId,
|
||||||
u_time: { value: options.u_time || 0 },
|
u_activeColor: options.activeColor
|
||||||
u_zoom: { value: options.u_zoom },
|
})),
|
||||||
u_duration: { value: options.u_duration || 2.0 },
|
defines,
|
||||||
u_interval: { value: options.u_interval || 1.0 },
|
|
||||||
u_trailLength: { value: options.u_trailLength || 0.2 },
|
|
||||||
u_activeId: { value: options.activeId || 0 },
|
|
||||||
u_activeColor: { value: options.activeColor || [ 1.0, 0, 0, 1.0 ] }
|
|
||||||
},
|
|
||||||
vertexShader: vs,
|
vertexShader: vs,
|
||||||
fragmentShader: fs,
|
fragmentShader: fs,
|
||||||
transparent: true,
|
transparent: true,
|
||||||
|
@ -60,23 +55,3 @@ export function MeshLineMaterial(options) {
|
||||||
});
|
});
|
||||||
return material;
|
return material;
|
||||||
}
|
}
|
||||||
export function DashLineMaterial(options) {
|
|
||||||
const { vs, fs } = getModule('meshline');
|
|
||||||
const material = new Material({
|
|
||||||
uniforms: {
|
|
||||||
u_opacity: { value: options.u_opacity || 1.0 },
|
|
||||||
u_time: { value: options.u_time || 0 },
|
|
||||||
u_zoom: { value: options.u_zoom },
|
|
||||||
u_dashSteps: { value: options.u_dashSteps || 12 },
|
|
||||||
u_dashSmooth: { value: options.u_dashSmooth || 0.01 },
|
|
||||||
u_dashDistance: { value: options.u_dashDistance || 0.2 },
|
|
||||||
u_activeId: { value: options.activeId || 0 },
|
|
||||||
u_activeColor: { value: options.activeColor || [ 1.0, 0, 0, 1.0 ] }
|
|
||||||
},
|
|
||||||
vertexShader: vs,
|
|
||||||
fragmentShader: fs,
|
|
||||||
transparent: true
|
|
||||||
});
|
|
||||||
return material;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
|
@ -1,10 +0,0 @@
|
||||||
varying float v_lineU;
|
|
||||||
uniform float u_dashSteps;
|
|
||||||
uniform float u_dashSmooth;
|
|
||||||
uniform float u_dashDistance;
|
|
||||||
varying vec4 v_color;
|
|
||||||
void main() {
|
|
||||||
float lineUMod = mod(v_lineU, 1.0/ u_dashSteps) * u_dashSteps;
|
|
||||||
float dash = smoothstep(u_dashDistance, u_dashDistance+u_dashSmooth, length(lineUMod-0.5));
|
|
||||||
gl_FragColor = vec4(v_color.xyz * vec3(dash), v_color.a * dash);
|
|
||||||
}
|
|
|
@ -1,23 +0,0 @@
|
||||||
precision highp float;
|
|
||||||
attribute float a_miter;
|
|
||||||
attribute vec4 a_color;
|
|
||||||
attribute float a_size;
|
|
||||||
uniform float u_zoom;
|
|
||||||
uniform float u_opacity;
|
|
||||||
varying vec4 v_color;
|
|
||||||
attribute float a_distance;
|
|
||||||
varying float v_lineU;
|
|
||||||
uniform float u_activeId;
|
|
||||||
uniform vec4 u_activeColor;
|
|
||||||
void main() {
|
|
||||||
v_lineU = a_distance;
|
|
||||||
mat4 matModelViewProjection = projectionMatrix * modelViewMatrix;
|
|
||||||
vec3 pointPos = position.xyz + vec3(normal * a_size * pow(2.0,20.0-u_zoom) / 2.0 * a_miter);
|
|
||||||
v_color = a_color;
|
|
||||||
v_color.a *= u_opacity;
|
|
||||||
if(pickingId == u_activeId) {
|
|
||||||
v_color = u_activeColor;
|
|
||||||
}
|
|
||||||
gl_Position = matModelViewProjection * vec4(pointPos, 1.0);
|
|
||||||
worldId = id_toPickColor(pickingId);
|
|
||||||
}
|
|
|
@ -19,10 +19,6 @@ import mesh_line_vert from '../shader/meshline_vert.glsl';
|
||||||
import line_frag from '../shader/line_frag.glsl';
|
import line_frag from '../shader/line_frag.glsl';
|
||||||
import line_vert from '../shader/line_vert.glsl';
|
import line_vert from '../shader/line_vert.glsl';
|
||||||
|
|
||||||
// 虚线
|
|
||||||
import dash_vert from '../shader/dashline_vert.glsl';
|
|
||||||
import dash_frag from '../shader/dashline_frag.glsl';
|
|
||||||
|
|
||||||
// 热力图
|
// 热力图
|
||||||
import heatmap_color_vert from '../shader/heatmap_colorize_vert.glsl';
|
import heatmap_color_vert from '../shader/heatmap_colorize_vert.glsl';
|
||||||
import heatmap_color_frag from '../shader/heatmap_colorize_frag.glsl';
|
import heatmap_color_frag from '../shader/heatmap_colorize_frag.glsl';
|
||||||
|
@ -58,7 +54,6 @@ export function compileBuiltinModules() {
|
||||||
registerModule('pointline', { vs: point_line_vert, fs: point_line_frag });
|
registerModule('pointline', { vs: point_line_vert, fs: point_line_frag });
|
||||||
registerModule('meshline', { vs: mesh_line_vert, fs: mesh_line_frag });
|
registerModule('meshline', { vs: mesh_line_vert, fs: mesh_line_frag });
|
||||||
registerModule('line', { vs: line_vert, fs: line_frag });
|
registerModule('line', { vs: line_vert, fs: line_frag });
|
||||||
registerModule('dashline', { vs: dash_vert, fs: dash_frag });
|
|
||||||
registerModule('heatmap_color', { vs: heatmap_color_vert, fs: heatmap_color_frag });
|
registerModule('heatmap_color', { vs: heatmap_color_vert, fs: heatmap_color_frag });
|
||||||
registerModule('heatmap_intensity', { vs: heatmap_intensity_vert, fs: heatmap_intensity_frag });
|
registerModule('heatmap_intensity', { vs: heatmap_intensity_vert, fs: heatmap_intensity_frag });
|
||||||
registerModule('text', { vs: text_vert, fs: text_frag });
|
registerModule('text', { vs: text_vert, fs: text_frag });
|
||||||
|
|
|
@ -1,11 +1,28 @@
|
||||||
precision highp float;
|
precision highp float;
|
||||||
uniform float u_opacity;
|
|
||||||
|
uniform float u_opacity : 1.0;
|
||||||
|
uniform float u_dash_offset : 0.0;
|
||||||
|
uniform float u_dash_ratio : 0.0;
|
||||||
|
uniform float u_blur : 0.9;
|
||||||
|
|
||||||
varying vec4 v_color;
|
varying vec4 v_color;
|
||||||
varying float vTime;
|
varying float v_distance;
|
||||||
|
varying float v_dash_array;
|
||||||
|
varying float v_time;
|
||||||
|
varying vec2 v_normal;
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
gl_FragColor = v_color;
|
gl_FragColor = v_color;
|
||||||
|
#ifdef DASHLINE
|
||||||
|
gl_FragColor.a *= u_opacity * ceil(mod(v_distance + u_dash_offset, v_dash_array) - (v_dash_array * u_dash_ratio));
|
||||||
|
#else
|
||||||
gl_FragColor.a = v_color.a * u_opacity;
|
gl_FragColor.a = v_color.a * u_opacity;
|
||||||
#ifdef ANIMATE
|
|
||||||
gl_FragColor.a *= vTime;
|
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef ANIMATE
|
||||||
|
gl_FragColor.a *= v_time;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// anti-alias
|
||||||
|
float blur = 1. - smoothstep(u_blur, 1., length(v_normal));
|
||||||
|
gl_FragColor.a *= blur;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,38 +1,48 @@
|
||||||
precision highp float;
|
|
||||||
attribute float a_miter;
|
attribute float a_miter;
|
||||||
attribute vec4 a_color;
|
attribute vec4 a_color;
|
||||||
attribute float a_size;
|
attribute float a_size;
|
||||||
attribute float a_distance;
|
attribute float a_distance;
|
||||||
|
attribute float a_dash_array;
|
||||||
|
|
||||||
uniform float u_zoom;
|
uniform float u_zoom;
|
||||||
|
uniform float u_time : 0;
|
||||||
|
uniform float u_activeId : 1;
|
||||||
|
uniform vec4 u_activeColor : [ 1.0, 0, 0, 1.0 ];
|
||||||
|
|
||||||
|
varying float v_time;
|
||||||
varying vec4 v_color;
|
varying vec4 v_color;
|
||||||
uniform float u_time;
|
varying float v_distance;
|
||||||
varying float vTime;
|
varying float v_dash_array;
|
||||||
uniform float u_activeId;
|
varying vec2 v_normal;
|
||||||
uniform vec4 u_activeColor;
|
|
||||||
// animate
|
|
||||||
#ifdef ANIMATE
|
#ifdef ANIMATE
|
||||||
uniform float u_duration; // 动画持续时间
|
uniform float u_duration : 2.0;
|
||||||
uniform float u_interval;
|
uniform float u_interval : 1.0;
|
||||||
uniform float u_repeat;
|
uniform float u_trailLength : 0.2;
|
||||||
uniform float u_trailLength;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
mat4 matModelViewProjection = projectionMatrix * modelViewMatrix;
|
|
||||||
vec3 pointPos = vec3(position.xy,0.) + vec3(normal * a_size * pow(2.0,20.0-u_zoom) / 2.0 * a_miter);
|
|
||||||
v_color = a_color;
|
v_color = a_color;
|
||||||
|
v_distance = a_distance;
|
||||||
|
v_dash_array = a_dash_array;
|
||||||
|
|
||||||
|
// anti-alias
|
||||||
|
v_normal = vec2(normal * sign(a_miter));
|
||||||
|
|
||||||
|
// extrude along normal
|
||||||
|
float extrude_scale = pow(2.0, 20.0 - u_zoom);
|
||||||
|
vec3 offset = vec3(normal * a_size * extrude_scale / 2.0 * a_miter);
|
||||||
|
gl_Position = projectionMatrix * modelViewMatrix * vec4(position.xy + offset.xy, 0., 1.0);
|
||||||
|
|
||||||
|
#ifdef ANIMATE
|
||||||
|
float alpha =1.0 - fract( mod(1.0- a_distance,u_interval)* (1.0/u_interval) + u_time / u_duration);
|
||||||
|
alpha = (alpha + u_trailLength -1.0) / u_trailLength;
|
||||||
|
v_time = clamp(alpha,0.,1.);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// picking
|
||||||
if(pickingId == u_activeId) {
|
if(pickingId == u_activeId) {
|
||||||
v_color = u_activeColor;
|
v_color = u_activeColor;
|
||||||
}
|
}
|
||||||
#ifdef ANIMATE
|
|
||||||
//mod(a_distance,0.2) * 5.
|
|
||||||
float alpa =1.0 - fract( mod(1.0- a_distance,u_interval)* (1.0/u_interval) + u_time / u_duration);
|
|
||||||
alpa = (alpa + u_trailLength -1.0) / u_trailLength;
|
|
||||||
vTime = clamp(alpa,0.,1.);
|
|
||||||
// vTime = (28800. + mod(u_time* 1000.,28800.)- position.z) / 100.;
|
|
||||||
#endif
|
|
||||||
worldId = id_toPickColor(pickingId);
|
worldId = id_toPickColor(pickingId);
|
||||||
gl_Position = matModelViewProjection * vec4(pointPos.xy, 0., 1.0);
|
|
||||||
|
|
||||||
}
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
import getNormal from 'polyline-normals';
|
import getNormals from '../../util/polyline-normals';
|
||||||
|
import flatten from '@antv/util/lib/flatten';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* shape arc
|
* shape arc
|
||||||
|
@ -64,69 +65,61 @@ export function defaultLine(geo, index) {
|
||||||
return { positions, indexes: indexArray };
|
return { positions, indexes: indexArray };
|
||||||
}
|
}
|
||||||
// mesh line
|
// mesh line
|
||||||
export function Line(path, props, positionsIndex) {
|
export function Line(path, props, positionsIndex, lengthPerDashSegment = 200) {
|
||||||
if (path.length === 1) path = path[0];// 面坐标转线坐标
|
if (path.length === 1) path = path[0];// 面坐标转线坐标
|
||||||
const positions = [];
|
const positions = [];
|
||||||
const pickingIds = [];
|
const pickingIds = [];
|
||||||
const normal = [];
|
const normal = [];
|
||||||
const miter = [];
|
const miter = [];
|
||||||
const colors = [];
|
const colors = [];
|
||||||
const indexArray = [];
|
const dashArray = [];
|
||||||
const normals = getNormal(path);
|
|
||||||
|
const { normals, attrIndex, attrPos } = getNormals(path, false, positionsIndex);
|
||||||
|
|
||||||
let attrDistance = [];
|
let attrDistance = [];
|
||||||
const sizes = [];
|
const sizes = [];
|
||||||
let c = 0;
|
|
||||||
let index = positionsIndex;
|
|
||||||
const { size, color, id } = props;
|
const { size, color, id } = props;
|
||||||
path.forEach((point, pointIndex, list) => {
|
attrPos.forEach((point, pointIndex) => {
|
||||||
const i = index;
|
|
||||||
colors.push(...color);
|
|
||||||
colors.push(...color);
|
colors.push(...color);
|
||||||
pickingIds.push(id);
|
pickingIds.push(id);
|
||||||
pickingIds.push(id);
|
|
||||||
sizes.push(size[0]);
|
sizes.push(size[0]);
|
||||||
sizes.push(size[0]);
|
point[2] = size[1];
|
||||||
if (pointIndex !== list.length - 1) {
|
|
||||||
indexArray[c++] = i + 0;
|
|
||||||
indexArray[c++] = i + 3;
|
|
||||||
indexArray[c++] = i + 1;
|
|
||||||
indexArray[c++] = i + 0;
|
|
||||||
indexArray[c++] = i + 2;
|
|
||||||
indexArray[c++] = i + 3;
|
|
||||||
}
|
|
||||||
// point[2] = size[1];
|
|
||||||
positions.push(...point);
|
|
||||||
positions.push(...point);
|
positions.push(...point);
|
||||||
|
|
||||||
if (pointIndex === 0) {
|
if (pointIndex === 0 || pointIndex === 1) {
|
||||||
attrDistance.push(0, 0);
|
attrDistance.push(0);
|
||||||
|
} else if (pointIndex % 2 === 0) {
|
||||||
|
attrDistance.push(attrDistance[pointIndex - 2]
|
||||||
|
+ lineSegmentDistance(attrPos[pointIndex - 2], attrPos[pointIndex]));
|
||||||
} else {
|
} else {
|
||||||
const d = attrDistance[pointIndex * 2 - 1] + lineSegmentDistance(path[pointIndex - 1], path[pointIndex]);
|
attrDistance.push(attrDistance[pointIndex - 1]);
|
||||||
attrDistance.push(d, d);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
index += 2;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const totalLength = attrDistance[attrDistance.length - 1];
|
||||||
|
const ratio = lengthPerDashSegment / totalLength;
|
||||||
normals.forEach(n => {
|
normals.forEach(n => {
|
||||||
const norm = n[0];
|
const norm = n[0];
|
||||||
const m = n[1];
|
const m = n[1];
|
||||||
normal.push(norm[0], norm[1], 0);
|
normal.push(norm[0], norm[1], 0);
|
||||||
normal.push(norm[0], norm[1], 0);
|
|
||||||
miter.push(-m);
|
|
||||||
miter.push(m);
|
miter.push(m);
|
||||||
|
dashArray.push(ratio);
|
||||||
});
|
});
|
||||||
|
|
||||||
attrDistance = attrDistance.map(d => {
|
attrDistance = attrDistance.map(d => {
|
||||||
return d / attrDistance[attrDistance.length - 1];
|
return d / totalLength;
|
||||||
});
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
positions,
|
positions,
|
||||||
normal,
|
normal,
|
||||||
indexArray,
|
indexArray: flatten(attrIndex),
|
||||||
miter,
|
miter,
|
||||||
colors,
|
colors,
|
||||||
sizes,
|
sizes,
|
||||||
pickingIds,
|
pickingIds,
|
||||||
attrDistance
|
attrDistance,
|
||||||
|
dashArray
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,15 +11,19 @@ export default function DrawLine(attributes, cfg, layer) {
|
||||||
geometry.addAttribute('normal', new THREE.Float32BufferAttribute(attributes.normal, 3));
|
geometry.addAttribute('normal', new THREE.Float32BufferAttribute(attributes.normal, 3));
|
||||||
geometry.addAttribute('a_miter', new THREE.Float32BufferAttribute(attributes.miter, 1));
|
geometry.addAttribute('a_miter', new THREE.Float32BufferAttribute(attributes.miter, 1));
|
||||||
geometry.addAttribute('a_distance', new THREE.Float32BufferAttribute(attributes.attrDistance, 1));
|
geometry.addAttribute('a_distance', new THREE.Float32BufferAttribute(attributes.attrDistance, 1));
|
||||||
|
geometry.addAttribute('a_dash_array', new THREE.Float32BufferAttribute(attributes.attrDashArray, 1));
|
||||||
|
|
||||||
const lineMaterial = new MeshLineMaterial({
|
const lineMaterial = new MeshLineMaterial({
|
||||||
u_opacity: style.opacity,
|
u_opacity: style.opacity,
|
||||||
u_zoom: zoom,
|
u_zoom: zoom,
|
||||||
u_time: 0,
|
u_time: 0,
|
||||||
|
u_dash_offset: style.dashOffset,
|
||||||
|
u_dash_ratio: style.dashRatio,
|
||||||
activeColor: activeOption.fill
|
activeColor: activeOption.fill
|
||||||
}, {
|
}, {
|
||||||
SHAPE: false,
|
SHAPE: false,
|
||||||
ANIMATE: false
|
ANIMATE: false,
|
||||||
|
DASHLINE: style.lineType === 'dash'
|
||||||
});
|
});
|
||||||
|
|
||||||
const lineMesh = new THREE.Mesh(geometry, lineMaterial);
|
const lineMesh = new THREE.Mesh(geometry, lineMaterial);
|
||||||
|
|
|
@ -16,6 +16,17 @@ export default function DrawFill(attributes, style) {
|
||||||
geometry.addAttribute('normal', new THREE.Float32BufferAttribute(attributes.normals, 3));
|
geometry.addAttribute('normal', new THREE.Float32BufferAttribute(attributes.normals, 3));
|
||||||
geometry.addAttribute('a_shape', new THREE.Float32BufferAttribute(attributes.shapePositions, 3));
|
geometry.addAttribute('a_shape', new THREE.Float32BufferAttribute(attributes.shapePositions, 3));
|
||||||
geometry.addAttribute('a_size', new THREE.Float32BufferAttribute(attributes.a_size, 3));
|
geometry.addAttribute('a_size', new THREE.Float32BufferAttribute(attributes.a_size, 3));
|
||||||
|
|
||||||
|
// const instancedGeometry = new THREE.InstancedBufferGeometry();
|
||||||
|
|
||||||
|
// instancedGeometry.addAttribute('normal', new THREE.Float32BufferAttribute(attributes.normals, 3));
|
||||||
|
// instancedGeometry.addAttribute('a_shape', new THREE.Float32BufferAttribute(attributes.shapePositions, 3));
|
||||||
|
// // instanced attributes
|
||||||
|
// instancedGeometry.addAttribute('position', new THREE.InstancedBufferAttribute(new Float32Array(attributes.vertices), 3));
|
||||||
|
// instancedGeometry.addAttribute('a_color', new THREE.InstancedBufferAttribute(new Float32Array(attributes.colors), 4));
|
||||||
|
// instancedGeometry.addAttribute('pickingId', new THREE.InstancedBufferAttribute(new Float32Array(attributes.pickingIds), 1));
|
||||||
|
// instancedGeometry.addAttribute('a_size', new THREE.InstancedBufferAttribute(new Float32Array(attributes.a_size), 3));
|
||||||
|
|
||||||
const material = new PolygonMaterial({
|
const material = new PolygonMaterial({
|
||||||
u_opacity: opacity,
|
u_opacity: opacity,
|
||||||
u_activeColor: activeColor
|
u_activeColor: activeColor
|
||||||
|
@ -25,6 +36,7 @@ export default function DrawFill(attributes, style) {
|
||||||
material.setDefinesvalue('SHAPE', true);
|
material.setDefinesvalue('SHAPE', true);
|
||||||
material.depthTest = false;
|
material.depthTest = false;
|
||||||
const fillMesh = new THREE.Mesh(geometry, material);
|
const fillMesh = new THREE.Mesh(geometry, material);
|
||||||
|
// const fillMesh = new THREE.Mesh(instancedGeometry, material);
|
||||||
return fillMesh;
|
return fillMesh;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,131 @@
|
||||||
|
/**
|
||||||
|
* 对于 polyline-normal 的改进
|
||||||
|
* 超过阈值,miter 转成 bevel 接头,
|
||||||
|
* 要注意 Three.js 中默认 THREE.FrontFaceDirectionCCW
|
||||||
|
* @see https://zhuanlan.zhihu.com/p/59541559
|
||||||
|
*/
|
||||||
|
import { direction, normal, computeMiter } from 'polyline-miter-util';
|
||||||
|
import { create, copy, dot } from 'gl-vec2';
|
||||||
|
|
||||||
|
function extrusions(positions, out, point, normal, scale) {
|
||||||
|
addNext(out, normal, -scale);
|
||||||
|
addNext(out, normal, scale);
|
||||||
|
positions.push(point);
|
||||||
|
positions.push(point);
|
||||||
|
}
|
||||||
|
|
||||||
|
function addNext(out, normal, length) {
|
||||||
|
out.push([[ normal[0], normal[1] ], length ]);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function(points, closed, indexOffset) {
|
||||||
|
const lineA = [ 0, 0 ];
|
||||||
|
const lineB = [ 0, 0 ];
|
||||||
|
const tangent = [ 0, 0 ];
|
||||||
|
const miter = [ 0, 0 ];
|
||||||
|
let _lastFlip = -1;
|
||||||
|
let _started = false;
|
||||||
|
let _normal = null;
|
||||||
|
const tmp = create();
|
||||||
|
let count = indexOffset || 0;
|
||||||
|
const miterLimit = 3;
|
||||||
|
|
||||||
|
const out = [];
|
||||||
|
const attrPos = [];
|
||||||
|
const attrIndex = [];
|
||||||
|
const attrCounters = [ 0, 0 ];
|
||||||
|
if (closed) {
|
||||||
|
points = points.slice();
|
||||||
|
points.push(points[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
const total = points.length;
|
||||||
|
|
||||||
|
for (let i = 1; i < total; i++) {
|
||||||
|
const index = count;
|
||||||
|
const last = points[i - 1];
|
||||||
|
const cur = points[i];
|
||||||
|
const next = i < points.length - 1 ? points[i + 1] : null;
|
||||||
|
|
||||||
|
attrCounters.push(i / total, i / total);
|
||||||
|
|
||||||
|
direction(lineA, cur, last);
|
||||||
|
|
||||||
|
if (!_normal) {
|
||||||
|
_normal = [ 0, 0 ];
|
||||||
|
normal(_normal, lineA);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!_started) {
|
||||||
|
_started = true;
|
||||||
|
extrusions(attrPos, out, last, _normal, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
attrIndex.push([ index + 0, index + 3, index + 1 ]);
|
||||||
|
|
||||||
|
// no miter, simple segment
|
||||||
|
if (!next) {
|
||||||
|
// reset normal
|
||||||
|
normal(_normal, lineA);
|
||||||
|
extrusions(attrPos, out, cur, _normal, 1);
|
||||||
|
attrIndex.push(
|
||||||
|
_lastFlip === 1 ? [ index + 1, index + 3, index + 2 ] : [ index, index + 2, index + 3 ]);
|
||||||
|
|
||||||
|
count += 2;
|
||||||
|
} else {
|
||||||
|
// get unit dir of next line
|
||||||
|
direction(lineB, next, cur);
|
||||||
|
|
||||||
|
// stores tangent & miter
|
||||||
|
let miterLen = computeMiter(tangent, miter, lineA, lineB, 1);
|
||||||
|
|
||||||
|
// get orientation
|
||||||
|
let flip = (dot(tangent, _normal) < 0) ? -1 : 1;
|
||||||
|
const bevel = miterLen > miterLimit;
|
||||||
|
if (bevel) {
|
||||||
|
miterLen = miterLimit;
|
||||||
|
attrCounters.push(i / total);
|
||||||
|
|
||||||
|
// next two points in our first segment
|
||||||
|
addNext(out, _normal, -flip);
|
||||||
|
attrPos.push(cur);
|
||||||
|
addNext(out, miter, miterLen * flip);
|
||||||
|
attrPos.push(cur);
|
||||||
|
|
||||||
|
attrIndex.push(_lastFlip !== -flip
|
||||||
|
? [ index + 1, index + 3, index + 2 ] : [ index, index + 2, index + 3 ]);
|
||||||
|
|
||||||
|
// now add the bevel triangle
|
||||||
|
attrIndex.push([ index + 2, index + 3, index + 4 ]);
|
||||||
|
|
||||||
|
normal(tmp, lineB);
|
||||||
|
copy(_normal, tmp); // store normal for next round
|
||||||
|
|
||||||
|
addNext(out, _normal, -flip);
|
||||||
|
attrPos.push(cur);
|
||||||
|
|
||||||
|
// the miter is now the normal for our next join
|
||||||
|
count += 3;
|
||||||
|
} else {
|
||||||
|
// next two points for our miter join
|
||||||
|
extrusions(attrPos, out, cur, miter, miterLen);
|
||||||
|
attrIndex.push(_lastFlip === 1
|
||||||
|
? [ index + 1, index + 3, index + 2 ] : [ index, index + 2, index + 3 ]);
|
||||||
|
|
||||||
|
flip = -1;
|
||||||
|
|
||||||
|
// the miter is now the normal for our next join
|
||||||
|
copy(_normal, miter);
|
||||||
|
count += 2;
|
||||||
|
}
|
||||||
|
_lastFlip = flip;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
normals: out,
|
||||||
|
attrIndex,
|
||||||
|
attrPos,
|
||||||
|
attrCounters
|
||||||
|
};
|
||||||
|
}
|
|
@ -1,3 +1,7 @@
|
||||||
|
import uniq from '@antv/util/lib/uniq';
|
||||||
|
import isString from '@antv/util/lib/is-string';
|
||||||
|
import ColorUtil from '../attr/color-util';
|
||||||
|
|
||||||
const SHADER_TYPE = {
|
const SHADER_TYPE = {
|
||||||
VS: 'vs',
|
VS: 'vs',
|
||||||
FS: 'fs'
|
FS: 'fs'
|
||||||
|
@ -9,9 +13,10 @@ const globalDefaultprecision = '#ifdef GL_FRAGMENT_PRECISION_HIGH\n precision hi
|
||||||
const globalDefaultAttribute = 'attribute float pickingId;\n varying vec4 worldId;\n';
|
const globalDefaultAttribute = 'attribute float pickingId;\n varying vec4 worldId;\n';
|
||||||
const globalDefaultInclude = '#pragma include "pick_color"\n';
|
const globalDefaultInclude = '#pragma include "pick_color"\n';
|
||||||
const includeRegExp = /#pragma include (["^+"]?["\ "[a-zA-Z_0-9](.*)"]*?)/g;
|
const includeRegExp = /#pragma include (["^+"]?["\ "[a-zA-Z_0-9](.*)"]*?)/g;
|
||||||
|
const uniformRegExp = /uniform\s+(bool|float|int|vec2|vec3|vec4|ivec2|ivec3|ivec4|mat2|mat3|mat4|sampler2D|samplerCube)\s+([\s\S]*?);/g;
|
||||||
|
|
||||||
function processModule(rawContent, includeList, type) {
|
function processModule(rawContent, includeList, type) {
|
||||||
return rawContent.replace(includeRegExp, (_, strMatch) => {
|
const compiled = rawContent.replace(includeRegExp, (_, strMatch) => {
|
||||||
const includeOpt = strMatch.split(' ');
|
const includeOpt = strMatch.split(' ');
|
||||||
const includeName = includeOpt[0].replace(/"/g, '');
|
const includeName = includeOpt[0].replace(/"/g, '');
|
||||||
|
|
||||||
|
@ -19,18 +24,110 @@ function processModule(rawContent, includeList, type) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
|
||||||
let txt = rawContentCache[includeName][type];
|
const txt = rawContentCache[includeName][type];
|
||||||
includeList.push(includeName);
|
includeList.push(includeName);
|
||||||
|
|
||||||
txt = processModule(txt, includeList, type);
|
const { content } = processModule(txt, includeList, type);
|
||||||
return txt;
|
return content;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
content: compiled,
|
||||||
|
includeList
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export function registerModule(moduleName, { vs, fs }) {
|
function getUniformLengthByType(type) {
|
||||||
|
let arrayLength = 0;
|
||||||
|
switch (type) {
|
||||||
|
case 'vec2':
|
||||||
|
case 'ivec2':
|
||||||
|
arrayLength = 2;
|
||||||
|
break;
|
||||||
|
case 'vec3':
|
||||||
|
case 'ivec3':
|
||||||
|
arrayLength = 3;
|
||||||
|
break;
|
||||||
|
case 'vec4':
|
||||||
|
case 'ivec4':
|
||||||
|
case 'mat2':
|
||||||
|
arrayLength = 4;
|
||||||
|
break;
|
||||||
|
case 'mat3':
|
||||||
|
arrayLength = 9;
|
||||||
|
break;
|
||||||
|
case 'mat4':
|
||||||
|
arrayLength = 16;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
return arrayLength;
|
||||||
|
}
|
||||||
|
|
||||||
|
function extractUniforms(content) {
|
||||||
|
const uniforms = {};
|
||||||
|
content = content.replace(uniformRegExp, (_, type, c) => {
|
||||||
|
const defaultValues = c.split(':');
|
||||||
|
const uniformName = defaultValues[0].trim();
|
||||||
|
let defaultValue = '';
|
||||||
|
if (defaultValues.length > 1) {
|
||||||
|
defaultValue = defaultValues[1].trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
// set default value for uniform according to its type
|
||||||
|
// eg. vec2 u -> [0.0, 0.0]
|
||||||
|
switch (type) {
|
||||||
|
case 'bool':
|
||||||
|
defaultValue = defaultValue === 'true';
|
||||||
|
break;
|
||||||
|
case 'float':
|
||||||
|
case 'int':
|
||||||
|
defaultValue = Number(defaultValue);
|
||||||
|
break;
|
||||||
|
case 'vec2':
|
||||||
|
case 'vec3':
|
||||||
|
case 'vec4':
|
||||||
|
case 'ivec2':
|
||||||
|
case 'ivec3':
|
||||||
|
case 'ivec4':
|
||||||
|
case 'mat2':
|
||||||
|
case 'mat3':
|
||||||
|
case 'mat4':
|
||||||
|
if (defaultValue) {
|
||||||
|
defaultValue = defaultValue.replace('[', '').replace(']', '')
|
||||||
|
.split(',')
|
||||||
|
.reduce((prev, cur) => {
|
||||||
|
prev.push(Number(cur.trim()));
|
||||||
|
return prev;
|
||||||
|
}, []);
|
||||||
|
} else {
|
||||||
|
defaultValue = new Array(getUniformLengthByType(type)).fill(0);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
uniforms[uniformName] = defaultValue;
|
||||||
|
return `uniform ${type} ${uniformName};\n`;
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
content,
|
||||||
|
uniforms
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function registerModule(moduleName, { vs, fs, uniforms: declaredUniforms }) {
|
||||||
|
const { content: extractedVS, uniforms: vsUniforms } = extractUniforms(vs);
|
||||||
|
const { content: extractedFS, uniforms: fsUniforms } = extractUniforms(fs);
|
||||||
|
|
||||||
rawContentCache[moduleName] = {
|
rawContentCache[moduleName] = {
|
||||||
[SHADER_TYPE.VS]: vs,
|
[SHADER_TYPE.VS]: extractedVS,
|
||||||
[SHADER_TYPE.FS]: fs
|
[SHADER_TYPE.FS]: extractedFS,
|
||||||
|
uniforms: {
|
||||||
|
...vsUniforms,
|
||||||
|
...fsUniforms,
|
||||||
|
...declaredUniforms
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,11 +136,20 @@ export function getModule(moduleName) {
|
||||||
return moduleCache[moduleName];
|
return moduleCache[moduleName];
|
||||||
}
|
}
|
||||||
|
|
||||||
let vs = rawContentCache[moduleName][SHADER_TYPE.VS];
|
let rawVS = rawContentCache[moduleName][SHADER_TYPE.VS];
|
||||||
let fs = rawContentCache[moduleName][SHADER_TYPE.FS];
|
const rawFS = rawContentCache[moduleName][SHADER_TYPE.FS];
|
||||||
vs = globalDefaultAttribute + globalDefaultInclude + vs;
|
|
||||||
vs = processModule(vs, [], SHADER_TYPE.VS);
|
rawVS = globalDefaultAttribute + globalDefaultInclude + rawVS;
|
||||||
fs = processModule(fs, [], SHADER_TYPE.FS);
|
|
||||||
|
const { content: vs, includeList: vsIncludeList } = processModule(rawVS, [], SHADER_TYPE.VS);
|
||||||
|
let { content: fs, includeList: fsIncludeList } = processModule(rawFS, [], SHADER_TYPE.FS);
|
||||||
|
// TODO: extract uniforms and their default values from GLSL
|
||||||
|
const uniforms = uniq(vsIncludeList.concat(fsIncludeList).concat(moduleName)).reduce((prev, cur) => {
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
...rawContentCache[cur].uniforms
|
||||||
|
};
|
||||||
|
}, {});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* set default precision for fragment shader
|
* set default precision for fragment shader
|
||||||
|
@ -55,7 +161,76 @@ export function getModule(moduleName) {
|
||||||
|
|
||||||
moduleCache[moduleName] = {
|
moduleCache[moduleName] = {
|
||||||
[SHADER_TYPE.VS]: vs.trim(),
|
[SHADER_TYPE.VS]: vs.trim(),
|
||||||
[SHADER_TYPE.FS]: fs.trim()
|
[SHADER_TYPE.FS]: fs.trim(),
|
||||||
|
uniforms
|
||||||
};
|
};
|
||||||
return moduleCache[moduleName];
|
return moduleCache[moduleName];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function wrapUniforms(uniforms) {
|
||||||
|
return Object.keys(uniforms).reduce((prev, cur) => {
|
||||||
|
prev[cur] = {
|
||||||
|
value: uniforms[cur]
|
||||||
|
};
|
||||||
|
return prev;
|
||||||
|
}, {});
|
||||||
|
}
|
||||||
|
|
||||||
|
const DEFAULT_LIGHT = {
|
||||||
|
type: 'directional',
|
||||||
|
direction: [ 1, 10.5, 12 ],
|
||||||
|
ambient: [ 0.2, 0.2, 0.2 ],
|
||||||
|
diffuse: [ 0.6, 0.6, 0.6 ],
|
||||||
|
specular: [ 0.1, 0.1, 0.1 ]
|
||||||
|
};
|
||||||
|
|
||||||
|
const DEFAULT_DIRECTIONAL_LIGHT = {
|
||||||
|
direction: [ 0, 0, 0 ],
|
||||||
|
ambient: [ 0, 0, 0 ],
|
||||||
|
diffuse: [ 0, 0, 0 ],
|
||||||
|
specular: [ 0, 0, 0 ]
|
||||||
|
};
|
||||||
|
|
||||||
|
const DEFAULT_SPOT_LIGHT = {
|
||||||
|
position: [ 0, 0, 0 ],
|
||||||
|
direction: [ 0, 0, 0 ],
|
||||||
|
ambient: [ 0, 0, 0 ],
|
||||||
|
diffuse: [ 0, 0, 0 ],
|
||||||
|
specular: [ 0, 0, 0 ],
|
||||||
|
constant: 1,
|
||||||
|
linear: 0,
|
||||||
|
quadratic: 0,
|
||||||
|
angle: 14,
|
||||||
|
exponent: 40,
|
||||||
|
blur: 5
|
||||||
|
};
|
||||||
|
|
||||||
|
const COLOR_ATTRIBUTES = [
|
||||||
|
'ambient', 'diffuse', 'specular'
|
||||||
|
];
|
||||||
|
|
||||||
|
export function generateLightingUniforms(lights) {
|
||||||
|
const lightsMap = {
|
||||||
|
u_directional_lights: new Array(3).fill({ ...DEFAULT_DIRECTIONAL_LIGHT }),
|
||||||
|
u_num_of_directional_lights: 0,
|
||||||
|
u_spot_lights: new Array(3).fill({ ...DEFAULT_SPOT_LIGHT }),
|
||||||
|
u_num_of_spot_lights: 0
|
||||||
|
};
|
||||||
|
if (!lights || !lights.length) {
|
||||||
|
lights = [ DEFAULT_LIGHT ];
|
||||||
|
}
|
||||||
|
lights.forEach(({ type, ...rest }, i) => {
|
||||||
|
const lightsUniformName = `u_${type}_lights`;
|
||||||
|
const lightsNumUniformName = `u_num_of_${type}_lights`;
|
||||||
|
|
||||||
|
Object.keys(rest).forEach(key => {
|
||||||
|
if (isString(rest[key]) && COLOR_ATTRIBUTES.indexOf(key) > -1) {
|
||||||
|
rest[key] = ColorUtil.color2RGBA(rest[key]).slice(0, 3);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
lightsMap[lightsUniformName][i] = { ...lightsMap[lightsUniformName][i], ...rest };
|
||||||
|
lightsMap[lightsNumUniformName]++;
|
||||||
|
});
|
||||||
|
return lightsMap;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue