add default particle props

This commit is contained in:
HifiExperiments 2024-04-17 15:43:20 -07:00
parent ac29616b7c
commit a6491d6225
5 changed files with 206 additions and 2 deletions

View file

@ -0,0 +1,7 @@
layout(location=2) in vec3 _normalWS;
float getProceduralFragment(inout ProceduralFragment proceduralData) {
proceduralData.normal = normalize(_normalWS);
proceduralData.diffuse = 0.5 * (proceduralData.normal + 1.0);
return 0.0;
}

View file

@ -0,0 +1,86 @@
uniform float radius = 0.01;
uniform float lifespan = 1.0; // seconds
layout(location=2) out vec3 _normalWS;
float bezierInterpolate(float y1, float y2, float y3, float u) {
// https://en.wikipedia.org/wiki/Bezier_curve
return (1.0 - u) * (1.0 - u) * y1 + 2.0 * (1.0 - u) * u * y2 + u * u * y3;
}
float interpolate3Points(float y1, float y2, float y3, float u) {
// Makes the interpolated values intersect the middle value.
if ((u <= 0.5f && y1 == y2) || (u >= 0.5f && y2 == y3)) {
// Flat line.
return y2;
}
float halfSlope;
if ((y2 >= y1 && y2 >= y3) || (y2 <= y1 && y2 <= y3)) {
// U or inverted-U shape.
// Make the slope at y2 = 0, which means that the control points half way between the value points have the value y2.
halfSlope = 0.0f;
} else {
// L or inverted and/or mirrored L shape.
// Make the slope at y2 be the slope between y1 and y3, up to a maximum of double the minimum of the slopes between y1
// and y2, and y2 and y3. Use this slope to calculate the control points half way between the value points.
// Note: The maximum ensures that the control points and therefore the interpolated values stay between y1 and y3.
halfSlope = (y3 - y1) / 2.0f;
float slope12 = y2 - y1;
float slope23 = y3 - y2;
{
float check = float(abs(halfSlope) > abs(slope12));
halfSlope = mix(halfSlope, slope12, check);
halfSlope = mix(halfSlope, slope23, (1.0 - check) * float(abs(halfSlope) > abs(slope23)));
}
}
float stepU = step(0.5f, u); // 0.0 if u < 0.5, 1.0 otherwise.
float slopeSign = 2.0f * stepU - 1.0f; // -1.0 if u < 0.5, 1.0 otherwise
float start = (1.0f - stepU) * y1 + stepU * y2; // y1 if u < 0.5, y2 otherwise
float middle = y2 + slopeSign * halfSlope;
float finish = (1.0f - stepU) * y2 + stepU * y3; // y2 if u < 0.5, y3 otherwise
float v = 2.0f * u - step(0.5f, u); // 0.0-0.5 -> 0.0-1.0 and 0.5-1.0 -> 0.0-1.0
return bezierInterpolate(start, middle, finish, v);
}
vec3 getProceduralVertex(const int particleID) {
vec4 positionAndAge = getParticleProperty(0, particleID);
vec3 position = positionAndAge.xyz;
const vec3 UP = vec3(0, 1, 0);
vec3 forward = normalize(getParticleProperty(1, particleID).xyz);
vec3 right = cross(forward, UP);
vec3 up = cross(right, forward);
const int VERTEX = gl_VertexID % 3;
int TRIANGLE = int(gl_VertexID / 3);
float age = positionAndAge.w;
float particleRadius = interpolate3Points(0.0, radius, 0.0, clamp(age / lifespan, 0.0, 1.0));
if (TRIANGLE < 3) {
const vec3 SIDE_POINTS[3] = vec3[3](
up,
normalize(-up + right),
normalize(-up - right)
);
position += particleRadius * (VERTEX == 2 ? forward : SIDE_POINTS[(TRIANGLE + VERTEX) % 3]);
_normalWS = normalize(cross(forward - SIDE_POINTS[TRIANGLE], forward - SIDE_POINTS[(TRIANGLE + 1) % 3]));
} else {
TRIANGLE -= 3;
vec3 backward = -2.0 * normalize(getParticleProperty(2, particleID).xyz);
const vec3 SIDE_POINTS[3] = vec3[3](
up,
normalize(-up - right),
normalize(-up + right)
);
position += particleRadius * (VERTEX == 2 ? backward : SIDE_POINTS[(TRIANGLE + VERTEX) % 3]);
_normalWS = normalize(cross(backward - SIDE_POINTS[TRIANGLE], backward - SIDE_POINTS[(TRIANGLE + 1) % 3]));
}
return position;
}

View file

@ -0,0 +1,73 @@
uniform float lifespan = 1.0; // seconds
uniform float speed = 0.1; // m/s
uniform float speedSpread = 0.25;
uniform float mass = 1.0;
const float G = 6.67e-11;
// prop0: xyz: position, w: age
// prop1: xyz: velocity, w: prevUpdateTime
// prop2: xyz: prevVelocity
vec3 initPosition(const int particleID) {
return 0.5 * (vec3(hifi_hash(particleID + iGlobalTime),
hifi_hash(particleID + iGlobalTime + 1.0),
hifi_hash(particleID + iGlobalTime + 2.0)) - 0.5);
}
mat3 rotationMatrix(vec3 axis, float angle) {
float s = sin(angle);
float c = cos(angle);
float oc = 1.0 - c;
return mat3(oc * axis.x * axis.x + c, oc * axis.x * axis.y - axis.z * s, oc * axis.z * axis.x + axis.y * s,
oc * axis.x * axis.y + axis.z * s, oc * axis.y * axis.y + c, oc * axis.y * axis.z - axis.x * s,
oc * axis.z * axis.x - axis.y * s, oc * axis.y * axis.z + axis.x * s, oc * axis.z * axis.z + c);
}
vec3 initVelocity(const int particleID, const vec3 position) {
const float particleSpeed = speed * ((1.0 - speedSpread) + speedSpread * hifi_hash(particleID + iGlobalTime + 3.0));
vec3 r = normalize(iWorldPosition - position);
float angle = 2.0 * 3.14159 * hifi_hash(particleID + iGlobalTime + 4.0);
return particleSpeed * rotationMatrix(r, angle) * cross(r, vec3(0, 1, 0));
}
void updateParticleProps(const int particleID, inout ParticleUpdateProps particleProps) {
// First draw
if (particleProps.prop1.w < 0.00001) {
particleProps.prop0.xyz = iWorldOrientation * (initPosition(particleID) * iWorldScale) + iWorldPosition;
particleProps.prop0.w = -lifespan * hifi_hash(particleID + iGlobalTime + 3.0);
particleProps.prop1.xyz = initVelocity(particleID, particleProps.prop0.xyz);
particleProps.prop1.w = iGlobalTime;
particleProps.prop2.xyz = particleProps.prop1.xyz;
return;
}
// Particle expired
if (particleProps.prop0.w >= lifespan) {
particleProps.prop0.xyz = iWorldOrientation * (initPosition(particleID) * iWorldScale) + iWorldPosition;
particleProps.prop0.w = 0.0;
particleProps.prop1.xyz = initVelocity(particleID, particleProps.prop0.xyz);
particleProps.prop1.w = iGlobalTime;
particleProps.prop2.xyz = particleProps.prop1.xyz;
return;
}
float dt = 0.01666666666;//max(0.0, iGlobalTime - particleProps.prop1.w);
particleProps.prop2.xyz = particleProps.prop1.xyz;
if (particleProps.prop0.w >= 0.0) {
// gravitational acceleration
vec3 r = iWorldPosition - particleProps.prop0.xyz;
vec3 g = (G * mass / max(0.01, dot(r, r))) * r;
// position
particleProps.prop0.xyz += particleProps.prop1.xyz * dt + (0.5 * dt * dt) * g;
// velocity
particleProps.prop1.xyz += g * dt;
}
// age
particleProps.prop0.w += dt;
// prevUpdateTime
particleProps.prop1.w = iGlobalTime;
}

View file

@ -1352,10 +1352,26 @@ EntityPropertyFlags EntityItemProperties::getChangedProperties() const {
* can manipulate the properties of, and use <code>JSON.stringify()</code> to convert the object into a string to put in
* the property.
*
* @example <caption>TODO</caption>
* @example <caption>A cube of oscillating, unlit, billboarded triangles, with the oscillation in the update (computed once per particle instead of once per vertex).</caption>
* particles = Entities.addEntity({
* type: "ProceduralParticleEffect",
* position: Vec3.sum(MyAvatar.position, Vec3.multiplyQbyV(MyAvatar.orientation, { x: 0, y: 0.5, z: -4 })),
* dimensions: 3,
* numParticles: 10000,
* numTrianglesPerParticle: 1,
* numUpdateProps: 1,
* particleUpdateData: JSON.stringify({
* version: 1.0,
* fragmentShaderURL: "https://gist.githubusercontent.com/HifiExperiments/9049fb4a8dcd2c1401ff4321103dce16/raw/4f9474ed82c66c1f94c1055d2724af808cd7aace/proceduralParticleUpdate.fs",
* }),
* particleRenderData: JSON.stringify({
* version: 1.0,
* vertexShaderURL: "https://gist.github.com/HifiExperiments/5dda24e28e7de1719e3a594d81306343/raw/92e0c5b82a9fa87685064cdbab92ed0c16f49f94/proceduralParticle2.vs",
* fragmentShaderURL: "https://gist.github.com/HifiExperiments/7def54504362c7bc79b5c85cd515b98b/raw/93b3828c2ec66b12b789a625dd141f533c595ede/proceduralParticle.fs",
* uniforms: {
* radius: 0.03
* }
* }),
* lifetime: 300 // Delete after 5 minutes.
* });
*/

View file

@ -484,7 +484,29 @@
azimuthFinish: Math.PI
},
ProceduralParticleEffect: {
// TODO: what should the default procedural particle be?
dimensions: 3,
numParticles: 10000,
numTrianglesPerParticle: 6,
numUpdateProps: 3,
particleUpdateData: JSON.stringify({
version: 1.0,
fragmentShaderURL: "qrc:///shaders/proceduralParticleSwarmUpdate.frag",
uniforms: {
lifespan: 3.0,
speed: 2.0,
speedSpread: 0.25,
mass: 50000000000
}
}),
particleRenderData: JSON.stringify({
version: 3.0,
vertexShaderURL: "qrc:///shaders/proceduralParticleSwarmRender.vert",
fragmentShaderURL: "qrc:///shaders/proceduralParticleSwarmRender.frag",
uniforms: {
radius: 0.03,
lifespan: 3.0
}
})
},
Light: {
color: { red: 255, green: 255, blue: 255 },