mirror of https://gitee.com/antv-l7/antv-l7
merge master
This commit is contained in:
commit
ea4d2bb447
|
@ -8,5 +8,9 @@ node_modules/
|
||||||
demos/assets/
|
demos/assets/
|
||||||
demos/index.html
|
demos/index.html
|
||||||
demos/*
|
demos/*
|
||||||
|
<<<<<<< HEAD
|
||||||
rollup/*
|
rollup/*
|
||||||
webpack/*
|
webpack/*
|
||||||
|
=======
|
||||||
|
src/core/three.js
|
||||||
|
>>>>>>> master
|
||||||
|
|
|
@ -1,15 +1,15 @@
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
babelrc: {
|
babelrc: {
|
||||||
presets: [
|
presets: [
|
||||||
"@babel/preset-env"
|
"@babel/preset-env"
|
||||||
]
|
],
|
||||||
},
|
},
|
||||||
extensions: ['.js'],
|
extensions: ['.js'],
|
||||||
include: [
|
include: [
|
||||||
'src/**/*.js',
|
'src/**/*.js',
|
||||||
'test/**/*.js',
|
'test/**/*.js',
|
||||||
|
'node_modules/_three@0.101.1@three/**/*.js',
|
||||||
'node_modules/three/**/*.js',
|
'node_modules/three/**/*.js',
|
||||||
'node_modules/simple-statistics/src/*.js'
|
'node_modules/simple-statistics/src/*.js'
|
||||||
],
|
],
|
||||||
|
|
|
@ -0,0 +1,76 @@
|
||||||
|
<!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>point_circle</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 scene = new L7.Scene({
|
||||||
|
id: 'map',
|
||||||
|
mapStyle: 'dark', // 样式URL
|
||||||
|
center: [120.132624,30.281774],
|
||||||
|
pitch: 0,
|
||||||
|
zoom: 10
|
||||||
|
});
|
||||||
|
scene.on('loaded', () => {
|
||||||
|
$.get('https://gw.alipayobjects.com/os/basement_prod/7359a5e9-3c5e-453f-b207-bc892fb23b84.csv', data => {
|
||||||
|
var layer = scene.HeatMapLayer({
|
||||||
|
zIndex: 2
|
||||||
|
})
|
||||||
|
.source(data, {
|
||||||
|
parser: {
|
||||||
|
type: 'csv',
|
||||||
|
x: 'lng',
|
||||||
|
y: 'lat'
|
||||||
|
},
|
||||||
|
transforms:[
|
||||||
|
{
|
||||||
|
type: 'map',
|
||||||
|
callback:function(item){
|
||||||
|
const [x, y] = item.coordinates;
|
||||||
|
item.lat = item.lat*1;
|
||||||
|
item.lng = item.lng*1;
|
||||||
|
item.v = item.v *1;
|
||||||
|
item.coordinates = [x*1,y*1];
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'hexagon',
|
||||||
|
size: 6000,
|
||||||
|
field:'v',
|
||||||
|
method:'sum'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
})
|
||||||
|
.active(true)
|
||||||
|
.shape('hexagon')
|
||||||
|
.style({
|
||||||
|
coverage: 0.9,
|
||||||
|
angle: 0,
|
||||||
|
})
|
||||||
|
.color('count', ["#002466","#105CB3","#2894E0","#CFF6FF","#FFF5B8","#FFAB5C","#F27049","#730D1C"])
|
||||||
|
.render();
|
||||||
|
console.log(layer);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
|
|
@ -78,7 +78,7 @@
|
||||||
"prepublishOnly": "npm run build-lib && npm run dist",
|
"prepublishOnly": "npm run build-lib && npm run dist",
|
||||||
"screenshot": "node ./bin/screenshot.js",
|
"screenshot": "node ./bin/screenshot.js",
|
||||||
"start": "npm run dev",
|
"start": "npm run dev",
|
||||||
"test": "torch --compile --renderer --recursive test/unit",
|
"test": "torch --compile-opts ./.torch.compile.opts.js --compile --renderer --recursive test/unit",
|
||||||
"test-all": "npm run test && npm run test-bugs",
|
"test-all": "npm run test && npm run test-bugs",
|
||||||
"test-bugs": "torch --compile --renderer --recursive test/bugs",
|
"test-bugs": "torch --compile --renderer --recursive test/bugs",
|
||||||
"test-bugs-live": "torch --compile --interactive --watch --recursive test/bugs",
|
"test-bugs-live": "torch --compile --interactive --watch --recursive test/bugs",
|
||||||
|
@ -109,7 +109,7 @@
|
||||||
"polyline-normals": "^2.0.2",
|
"polyline-normals": "^2.0.2",
|
||||||
"rbush": "^2.0.2",
|
"rbush": "^2.0.2",
|
||||||
"simple-statistics": "^7.0.1",
|
"simple-statistics": "^7.0.1",
|
||||||
"three": "^0.96.0",
|
"three": "^0.101.1",
|
||||||
"venn.js": "^0.2.20",
|
"venn.js": "^0.2.20",
|
||||||
"viewport-mercator-project": "^5.2.0",
|
"viewport-mercator-project": "^5.2.0",
|
||||||
"webworkify-webpack": "^2.1.3",
|
"webworkify-webpack": "^2.1.3",
|
||||||
|
|
|
@ -35,4 +35,4 @@ export {
|
||||||
BufferAttribute
|
BufferAttribute
|
||||||
} from 'three/src/core/BufferAttribute.js';
|
} from 'three/src/core/BufferAttribute.js';
|
||||||
|
|
||||||
// export * from '../../build/Three.js';
|
// export * from '../../build/three.js';
|
||||||
|
|
|
@ -0,0 +1,31 @@
|
||||||
|
import { fill } from '../../shape/polygon';
|
||||||
|
export default function hexagonBuffer(layerData) {
|
||||||
|
const attribute = {
|
||||||
|
vertices: [],
|
||||||
|
miter: [],
|
||||||
|
colors: [],
|
||||||
|
pickingIds: []
|
||||||
|
};
|
||||||
|
const a = Math.cos(Math.PI / 6);
|
||||||
|
const points = [
|
||||||
|
[ 0, -1, 0 ],
|
||||||
|
[ -a, -0.5, 0 ],
|
||||||
|
[ -a, 0.5, 0 ],
|
||||||
|
[ 0, 1, 0 ],
|
||||||
|
[ a, 0.5, 0 ],
|
||||||
|
[ a, -0.5, 0 ],
|
||||||
|
[ 0, -1, 0 ]
|
||||||
|
];
|
||||||
|
// const hexgonPoints = polygonPath(6);
|
||||||
|
const hexgonFill = fill([ points ]);
|
||||||
|
const { positionsIndex, positions } = hexgonFill;
|
||||||
|
layerData.forEach(element => {
|
||||||
|
positionsIndex.forEach(pointIndex => {
|
||||||
|
attribute.vertices.push(...element.coordinates);
|
||||||
|
attribute.miter.push(positions[pointIndex][0], positions[pointIndex][1]);
|
||||||
|
attribute.pickingIds.push(element.id);
|
||||||
|
attribute.colors.push(...element.color);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return attribute;
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
import grid_frag from '../shader/hexagon_frag.glsl';
|
||||||
|
import grid_vert from '../shader/hexagon_vert.glsl';
|
||||||
|
import Material from './material';
|
||||||
|
|
||||||
|
|
||||||
|
export default class hexagonMaterial extends Material {
|
||||||
|
getDefaultParameters() {
|
||||||
|
return {
|
||||||
|
uniforms: {
|
||||||
|
u_opacity: { value: 1.0 },
|
||||||
|
u_time: { value: 0 },
|
||||||
|
u_radius: { value: 0.01 },
|
||||||
|
u_angle: { value: 0.01 },
|
||||||
|
u_coverage: { value: 0.8 }
|
||||||
|
},
|
||||||
|
defines: {
|
||||||
|
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
constructor(_uniforms, _defines, parameters) {
|
||||||
|
super(parameters);
|
||||||
|
const { uniforms, defines } = this.getDefaultParameters();
|
||||||
|
this.uniforms = Object.assign(uniforms, this.setUniform(_uniforms));
|
||||||
|
this.type = 'hexagonMaterial';
|
||||||
|
this.defines = Object.assign(defines, _defines);
|
||||||
|
this.vertexShader = grid_vert;
|
||||||
|
this.fragmentShader = grid_frag;
|
||||||
|
this.transparent = true;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
precision highp float;
|
||||||
|
uniform float u_opacity;
|
||||||
|
varying vec4 v_color;
|
||||||
|
void main() {
|
||||||
|
vec4 color = v_color;
|
||||||
|
gl_FragColor = color;
|
||||||
|
gl_FragColor.a =color.a*u_opacity;
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
precision highp float;
|
||||||
|
attribute vec2 miter;
|
||||||
|
attribute vec4 a_color;
|
||||||
|
uniform float u_radius;
|
||||||
|
uniform float u_coverage;
|
||||||
|
uniform float u_angle;
|
||||||
|
varying vec4 v_color;
|
||||||
|
|
||||||
|
void main() {
|
||||||
|
mat4 matModelViewProjection = projectionMatrix * modelViewMatrix;
|
||||||
|
mat2 rotationMatrix = mat2(cos(u_angle), sin(u_angle), -sin(u_angle), cos(u_angle));
|
||||||
|
v_color = a_color;
|
||||||
|
vec2 offset =vec2(rotationMatrix * miter * u_radius * u_coverage );
|
||||||
|
float x = position.x + offset.x;
|
||||||
|
float y = position.y + offset.y;
|
||||||
|
gl_Position = matModelViewProjection * vec4(x, y, position.z, 1.0);
|
||||||
|
}
|
|
@ -2,7 +2,6 @@ precision highp float;
|
||||||
attribute vec4 a_color;
|
attribute vec4 a_color;
|
||||||
attribute float a_size;
|
attribute float a_size;
|
||||||
attribute float a_shape;
|
attribute float a_shape;
|
||||||
attribute vec4 a_idColor;
|
|
||||||
uniform vec4 u_stroke;
|
uniform vec4 u_stroke;
|
||||||
uniform float u_strokeWidth;
|
uniform float u_strokeWidth;
|
||||||
uniform float u_opacity;
|
uniform float u_opacity;
|
||||||
|
|
|
@ -3,7 +3,6 @@ precision highp float;
|
||||||
#define diffuseRatio 0.4
|
#define diffuseRatio 0.4
|
||||||
#define specularRatio 0.1
|
#define specularRatio 0.1
|
||||||
attribute vec4 a_color;
|
attribute vec4 a_color;
|
||||||
attribute vec4 a_idColor;
|
|
||||||
attribute vec2 faceUv;
|
attribute vec2 faceUv;
|
||||||
attribute vec3 a_shape;
|
attribute vec3 a_shape;
|
||||||
attribute vec3 a_size;
|
attribute vec3 a_size;
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
import Layer from '../core/layer';
|
import Layer from '../core/layer';
|
||||||
import gridBuffer from '../geom/buffer/heatmap/grid';
|
import gridBuffer from '../geom/buffer/heatmap/grid';
|
||||||
import DrawGrid from './render/heatmap/gird';
|
import DrawGrid from './render/heatmap/gird';
|
||||||
|
import DrawHexagon from './render/heatmap/hexagon';
|
||||||
|
import hexagonBuffer from '../geom/buffer/heatmap/hexagon';
|
||||||
|
|
||||||
export default class HeatMapLayer extends Layer {
|
export default class HeatMapLayer extends Layer {
|
||||||
shape(type) {
|
shape(type) {
|
||||||
|
@ -13,6 +15,29 @@ export default class HeatMapLayer extends Layer {
|
||||||
}
|
}
|
||||||
_prepareRender() {
|
_prepareRender() {
|
||||||
this.init();
|
this.init();
|
||||||
|
switch (this.shapeType) {
|
||||||
|
case 'grid' :
|
||||||
|
this._drawGrid();
|
||||||
|
break;
|
||||||
|
case 'hexagon' :
|
||||||
|
this._drawHexagon();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
this._drawGrid();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_drawHexagon() {
|
||||||
|
const style = this.get('styleOptions');
|
||||||
|
const { radius } = this.layerSource.data;
|
||||||
|
this._buffer = new hexagonBuffer(this.layerData);
|
||||||
|
const config = {
|
||||||
|
...style,
|
||||||
|
radius
|
||||||
|
};
|
||||||
|
const Mesh = new DrawHexagon(this._buffer, config);
|
||||||
|
this.add(Mesh);
|
||||||
|
}
|
||||||
|
_drawGrid() {
|
||||||
this.type = 'heatmap';
|
this.type = 'heatmap';
|
||||||
const style = this.get('styleOptions');
|
const style = this.get('styleOptions');
|
||||||
const { xOffset, yOffset } = this.layerSource.data;
|
const { xOffset, yOffset } = this.layerSource.data;
|
||||||
|
@ -25,4 +50,5 @@ export default class HeatMapLayer extends Layer {
|
||||||
const girdMesh = new DrawGrid(this._buffer, config);
|
const girdMesh = new DrawGrid(this._buffer, config);
|
||||||
this.add(girdMesh);
|
this.add(girdMesh);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,21 @@
|
||||||
|
import * as THREE from '../../../core/three';
|
||||||
|
import GridMaterial from '../../../geom/material/hexagon';
|
||||||
|
export default function DrawHexagon(attributes, style) {
|
||||||
|
const { opacity, radius, angle, coverage } = style;
|
||||||
|
const geometry = new THREE.BufferGeometry();
|
||||||
|
geometry.addAttribute('position', new THREE.Float32BufferAttribute(attributes.vertices, 3));
|
||||||
|
geometry.addAttribute('miter', new THREE.Float32BufferAttribute(attributes.miter, 2));
|
||||||
|
geometry.addAttribute('a_color', new THREE.Float32BufferAttribute(attributes.colors, 4));
|
||||||
|
geometry.addAttribute('pickingId', new THREE.Float32BufferAttribute(attributes.pickingIds, 1));
|
||||||
|
const material = new GridMaterial({
|
||||||
|
u_opacity: opacity,
|
||||||
|
u_radius: radius,
|
||||||
|
u_angle: angle / 180 * Math.PI,
|
||||||
|
u_coverage: coverage
|
||||||
|
}, {
|
||||||
|
SHAPE: false
|
||||||
|
});
|
||||||
|
const hexgonMesh = new THREE.Mesh(geometry, material);
|
||||||
|
return hexgonMesh;
|
||||||
|
}
|
||||||
|
|
|
@ -8,6 +8,7 @@ import raster from './parser/raster';
|
||||||
|
|
||||||
import { registerTransform, registerParser } from './factory';
|
import { registerTransform, registerParser } from './factory';
|
||||||
import { aggregatorToGrid } from './transform/grid';
|
import { aggregatorToGrid } from './transform/grid';
|
||||||
|
import { pointToHexbin } from './transform/hexagon';
|
||||||
import { map } from './transform/map';
|
import { map } from './transform/map';
|
||||||
|
|
||||||
registerParser('geojson', geojson);
|
registerParser('geojson', geojson);
|
||||||
|
@ -18,6 +19,7 @@ registerParser('raster', raster);
|
||||||
// 注册transform
|
// 注册transform
|
||||||
|
|
||||||
registerTransform('grid', aggregatorToGrid);
|
registerTransform('grid', aggregatorToGrid);
|
||||||
|
registerTransform('hexagon', pointToHexbin);
|
||||||
registerTransform('map', map);
|
registerTransform('map', map);
|
||||||
|
|
||||||
export { getTransform, registerTransform, getParser, registerParser } from './factory';
|
export { getTransform, registerTransform, getParser, registerParser } from './factory';
|
||||||
|
|
|
@ -11,8 +11,8 @@ export function aggregatorToGrid(data, option) {
|
||||||
const { gridHash, gridOffset } = _pointsGridHash(dataArray, size);
|
const { gridHash, gridOffset } = _pointsGridHash(dataArray, size);
|
||||||
const layerData = _getGridLayerDataFromGridHash(gridHash, gridOffset, option);
|
const layerData = _getGridLayerDataFromGridHash(gridHash, gridOffset, option);
|
||||||
return {
|
return {
|
||||||
xOffset: gridOffset.xOffset / 360 * (256 << 20) / 2,
|
|
||||||
yOffset: gridOffset.xOffset / 360 * (256 << 20) / 2,
|
yOffset: gridOffset.xOffset / 360 * (256 << 20) / 2,
|
||||||
|
xOffset: gridOffset.xOffset / 360 * (256 << 20) / 2,
|
||||||
dataArray: layerData
|
dataArray: layerData
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,7 +20,7 @@ export function pointToHexbin(data, option) {
|
||||||
.y(d => d.coordinates[1]);
|
.y(d => d.coordinates[1]);
|
||||||
const hexbinBins = newHexbin(screenPoints);
|
const hexbinBins = newHexbin(screenPoints);
|
||||||
const result = {
|
const result = {
|
||||||
size: pixlSize
|
radius: pixlSize
|
||||||
};
|
};
|
||||||
result.dataArray = hexbinBins.map((hex, index) => {
|
result.dataArray = hexbinBins.map((hex, index) => {
|
||||||
if (option.field && option.method) {
|
if (option.field && option.method) {
|
||||||
|
@ -31,8 +31,9 @@ export function pointToHexbin(data, option) {
|
||||||
item[option.method] = hex[option.method];
|
item[option.method] = hex[option.method];
|
||||||
return {
|
return {
|
||||||
...item,
|
...item,
|
||||||
|
count: hex.length,
|
||||||
coordinates: unProjectFlat([ hex.x, hex.y ]),
|
coordinates: unProjectFlat([ hex.x, hex.y ]),
|
||||||
id: index + 1
|
_id: index + 1
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -0,0 +1,35 @@
|
||||||
|
import { expect } from 'chai';
|
||||||
|
import hexagonBuffer from '../../../../../src/geom/buffer/heatmap/hexagon';
|
||||||
|
describe('hexagon heatMap buffer', () => {
|
||||||
|
const layerData = [
|
||||||
|
{ color: [ 1, 1, 0, 1 ], coordinates: [ 120.12063099925889, 30.263947783103486, 0 ], id: 1 },
|
||||||
|
{ color: [ 1, 0.5, 0, 1 ], coordinates: [ 120.12218696039365, 30.263947783103486, 0 ], id: 2 },
|
||||||
|
{
|
||||||
|
color: [ 1, 0.1, 0, 1 ],
|
||||||
|
coordinates: [ 120.12374292152843, 30.263947783103486, 0 ],
|
||||||
|
id: 3
|
||||||
|
},
|
||||||
|
{ color: [ 1, 0, 0.2, 1 ], coordinates: [ 120.1252988826632, 30.263947783103486, 0 ], id: 4 },
|
||||||
|
{
|
||||||
|
color: [ 0, 1, 0, 1 ],
|
||||||
|
coordinates: [ 120.12607686323062, 30.263947783103486, 0 ],
|
||||||
|
id: 5
|
||||||
|
},
|
||||||
|
{
|
||||||
|
color: [ 1, 1, 0, 1 ],
|
||||||
|
coordinates: [ 120.12685484379799, 30.263947783103486, 0 ],
|
||||||
|
id: 6
|
||||||
|
},
|
||||||
|
{ color: [ 1, 1, 1, 1 ], coordinates: [ 120.12841080493274, 30.263947783103486, 0 ], id: 7 },
|
||||||
|
{ color: [ 0, 1, 1, 1 ], coordinates: [ 120.13230070776972, 30.263947783103486, 0 ], id: 8 },
|
||||||
|
{ color: [ 0, 1, 0, 1 ], coordinates: [ 120.12763282436538, 30.263947783103486, 0 ], id: 9 },
|
||||||
|
{ color: [ 1, 1, 0, 1 ], coordinates: [ 120.12996676606754, 30.263947783103486, 0 ], id: 10 }
|
||||||
|
];
|
||||||
|
it('hexagon buffer', () => {
|
||||||
|
const attribute = hexagonBuffer(layerData);
|
||||||
|
expect(attribute.colors.length).eql(480);
|
||||||
|
expect(attribute.miter.length).eql(240);
|
||||||
|
expect(attribute.pickingIds.length).eql(120);
|
||||||
|
expect(attribute.vertices.length).eql(360);
|
||||||
|
});
|
||||||
|
});
|
|
@ -24,9 +24,8 @@ describe('test shader module', function() {
|
||||||
|
|
||||||
it('should import a module correctly.', function() {
|
it('should import a module correctly.', function() {
|
||||||
const { vs, fs } = getModule('module1');
|
const { vs, fs } = getModule('module1');
|
||||||
|
|
||||||
expect(vs).eq('#define PI 3.14');
|
expect(vs).eq('#define PI 3.14');
|
||||||
expect(fs).eq('');
|
expect(fs.replace(/(\s+)|(\n)+|(\r\n)+/g, '')).eqls('#ifdefGL_FRAGMENT_PRECISION_HIGHprecisionhighpfloat;#elseprecisionmediumpfloat;#endif');
|
||||||
});
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -4,7 +4,8 @@ const pkg = require('./package.json');
|
||||||
module.exports = {
|
module.exports = {
|
||||||
devtool: 'cheap-source-map',
|
devtool: 'cheap-source-map',
|
||||||
entry: {
|
entry: {
|
||||||
l7: './src/index.js'
|
l7: './src/index.js',
|
||||||
|
three: './src/core/three.js'
|
||||||
},
|
},
|
||||||
output: {
|
output: {
|
||||||
filename: '[name].js',
|
filename: '[name].js',
|
||||||
|
|
Loading…
Reference in New Issue