src/loaders/objects/tracks.ts
Custom mesh for tracks.
BufferGeometry
Properties |
Methods |
constructor()
|
Defined in src/loaders/objects/tracks.ts:35
|
Create the tracks mesh. |
_attributes |
Type : any
|
Defined in src/loaders/objects/tracks.ts:35
|
Attributes. |
colors |
Type : number[]
|
Defined in src/loaders/objects/tracks.ts:25
|
Colors. |
counter |
Type : number[]
|
Defined in src/loaders/objects/tracks.ts:27
|
Counter. |
indices_array |
Type : any[]
|
Defined in src/loaders/objects/tracks.ts:33
|
Indices. |
next |
Type : number[]
|
Defined in src/loaders/objects/tracks.ts:23
|
Next. |
next_track_id |
Type : number
|
Defined in src/loaders/objects/tracks.ts:17
|
ID of the next track. |
positions |
Type : number[]
|
Defined in src/loaders/objects/tracks.ts:19
|
Positions. |
previous |
Type : number[]
|
Defined in src/loaders/objects/tracks.ts:21
|
Previous. |
side |
Type : any[]
|
Defined in src/loaders/objects/tracks.ts:31
|
Side. |
track_id |
Type : any[]
|
Defined in src/loaders/objects/tracks.ts:29
|
Track ID. |
addTrack | ||||||||||||
addTrack(points: Vector3[], color: ColorRepresentation)
|
||||||||||||
Defined in src/loaders/objects/tracks.ts:60
|
||||||||||||
Add a track to the tracks mesh.
Parameters :
Returns :
number
ID of the track. |
process |
process()
|
Defined in src/loaders/objects/tracks.ts:102
|
Process the track mesh.
Returns :
void
|
import {
BufferGeometry,
BufferAttribute,
ShaderMaterial,
Vector2,
Vector3,
Color,
type ColorRepresentation,
type ShaderMaterialParameters,
} from 'three';
/**
* Custom mesh for tracks.
*/
export class TracksMesh extends BufferGeometry {
/** ID of the next track. */
next_track_id: number;
/** Positions. */
positions: number[];
/** Previous. */
previous: number[];
/** Next. */
next: number[];
/** Colors. */
colors: number[];
/** Counter. */
counter: number[];
/** Track ID. */
track_id: any[];
/** Side. */
side: any[];
/** Indices. */
indices_array: any[];
/** Attributes. */
_attributes: any;
/**
* Create the tracks mesh.
*/
constructor() {
super();
(this.type as string) = 'TracksMesh';
this.positions = [];
this.previous = [];
this.next = [];
this.side = [];
this.counter = [];
this.track_id = [];
this.colors = [];
this.indices_array = [];
this.next_track_id = 0;
}
/**
* Add a track to the tracks mesh.
* @param points Points for constructing the track.
* @param color Color of the track.
* @returns ID of the track.
*/
addTrack(points: Vector3[], color: ColorRepresentation) {
const id = this.next_track_id++;
const col = new Color(color);
this.previous.push(points[0].x, points[0].y, points[0].z);
this.previous.push(points[0].x, points[0].y, points[0].z);
for (let i = 0; i < points.length; i++) {
const p = points[i];
const n = this.positions.length / 3;
this.positions.push(p.x, p.y, p.z);
this.positions.push(p.x, p.y, p.z);
this.side.push(1, -1);
this.track_id.push(id, id);
this.colors.push(col.r, col.g, col.b);
this.colors.push(col.r, col.g, col.b);
this.counter.push(i / points.length, i / points.length);
if (i < points.length - 1) {
this.previous.push(p.x, p.y, p.z);
this.previous.push(p.x, p.y, p.z);
this.indices_array.push(n, n + 1, n + 2);
this.indices_array.push(n + 2, n + 1, n + 3);
} else if (i > 0) {
this.next.push(p.x, p.y, p.z);
this.next.push(p.x, p.y, p.z);
}
}
const i = points.length - 1;
this.next.push(points[i].x, points[i].y, points[i].z);
this.next.push(points[i].x, points[i].y, points[i].z);
//this.process()
return id;
}
/**
* Process the track mesh.
*/
process() {
if (
!this._attributes ||
this._attributes.position.count !== this.positions.length
) {
this._attributes = {
position: new BufferAttribute(new Float32Array(this.positions), 3),
previous: new BufferAttribute(new Float32Array(this.previous), 3),
next: new BufferAttribute(new Float32Array(this.next), 3),
side: new BufferAttribute(new Float32Array(this.side), 1),
track_id: new BufferAttribute(new Int32Array(this.track_id), 1),
color: new BufferAttribute(new Float32Array(this.colors), 3),
counter: new BufferAttribute(new Float32Array(this.counter), 1),
index: new BufferAttribute(new Uint32Array(this.indices_array), 1),
};
} else {
(this._attributes.position as BufferAttribute).copyArray(
new Float32Array(this.positions),
);
this._attributes.position.needsUpdate = true;
(this._attributes.previous as BufferAttribute).copyArray(
new Float32Array(this.previous),
);
this._attributes.previous.needsUpdate = true;
(this._attributes.next as BufferAttribute).copyArray(
new Float32Array(this.next),
);
this._attributes.next.needsUpdate = true;
(this._attributes.side as BufferAttribute).copyArray(
new Float32Array(this.side),
);
this._attributes.side.needsUpdate = true;
(this._attributes.track_id as BufferAttribute).copyArray(
new Int32Array(this.track_id),
);
this._attributes.track_id.needsUpdate = true;
(this._attributes.color as BufferAttribute).copyArray(
new Float32Array(this.colors),
);
this._attributes.color.needsUpdate = true;
(this._attributes.counter as BufferAttribute).copyArray(
new Float32Array(this.counter),
);
this._attributes.counter.needsUpdate = true;
(this._attributes.index as BufferAttribute).copyArray(
new Uint32Array(this.indices_array),
);
this._attributes.index.needsUpdate = true;
}
this.setAttribute('position', this._attributes.position);
this.setAttribute('previous', this._attributes.previous);
this.setAttribute('next', this._attributes.next);
this.setAttribute('side', this._attributes.side);
this.setAttribute('track_id', this._attributes.track_id);
this.setAttribute('color', this._attributes.color);
this.setAttribute('counter', this._attributes.counter);
this.setIndex(this._attributes.index as BufferAttribute);
this.computeBoundingSphere();
this.computeBoundingBox();
}
}
/** Custom vertex shader for tracks. */
const tracks_vert = [
'attribute vec3 previous;',
'attribute vec3 next;',
'attribute int track_id;',
'attribute float side;',
'attribute vec3 color;',
'attribute float counter;',
'varying vec3 v_color;',
'varying float v_counter;',
'flat varying int v_track_id;',
'uniform vec2 resolution;',
'uniform float lineWidth;',
'void main() {',
' vec2 aspect = vec2(resolution.x / resolution.y, 1.0);',
'',
' v_color = color;',
' v_counter = counter;',
' v_track_id = track_id;',
'',
' mat4 m = projectionMatrix * modelViewMatrix;',
' vec4 finalPosition = m * vec4(position, 1.0);',
' vec4 prevPos = m * vec4(previous, 1.0);',
' vec4 nextPos = m * vec4(next, 1.0);',
'',
' vec2 curP = finalPosition.xy / finalPosition.w * aspect;',
' vec2 prevP = prevPos.xy / prevPos.w * aspect;',
' vec2 nextP = nextPos.xy / nextPos.w * aspect;',
'',
' vec2 dir;',
' if (curP == prevP) dir = normalize(nextP - curP);',
' else if (curP == nextP) dir = normalize(curP - prevP);',
' else dir = normalize(curP - prevP);',
'',
' vec2 normal = vec2(-dir.y, dir.x);',
' normal.xy *= .5 * lineWidth;',
' normal.x /= aspect.x;',
' normal.xy *= finalPosition.w * 0.001;',
' finalPosition.xy += normal.xy * side;',
' gl_Position = finalPosition;',
'}',
].join('\n');
/** Custom fragment shadre for track. */
const tracks_frag = [
'uniform float progress;',
'varying vec3 v_color;',
'varying float v_counter;',
'flat varying int v_track_id;',
'void main() {',
' if (v_counter > progress) discard;',
' gl_FragColor = vec4(v_color, 1.0);',
'}',
].join('\n');
/**
* Custom material for the tracks.
*/
export class TracksMaterial extends ShaderMaterial {
/** If the material is of track. */
isTracksMaterial: boolean;
static get type() {
return 'TracksMaterial';
}
/**
* Create the tracks material.
* @param params Params for creating the tracks material.
*/
constructor(params: ShaderMaterialParameters) {
super({
uniforms: Object.assign(
{},
{
lineWidth: { value: 1 },
resolution: { value: new Vector2(1, 1) },
progress: { value: 1 },
},
),
vertexShader: tracks_vert,
fragmentShader: tracks_vert,
});
this.isTracksMaterial = true;
Object.defineProperties(this, {
lineWidth: {
enumerable: true,
get: function () {
return this.uniforms.lineWidth.value;
},
set: function (value) {
this.uniforms.lineWidth.value = value;
},
},
resolution: {
enumerable: true,
get: function () {
return this.uniforms.resolution.value;
},
set: function (value) {
this.uniforms.resolution.value.copy(value);
},
},
progress: {
enumerable: true,
get: function () {
return this.uniforms.progress.value;
},
set: function (value) {
this.uniforms.progress.value = value;
},
},
});
this.setValues(params);
}
}