Hash :
17cee572
Author :
Date :
2018-06-18T08:28:50
SampleApps: Refactor to use cli args the same way Bug: angleproject:2669 Change-Id: Ica5b49e63e8af69595cdde3737b0aa1808d7d841 Reviewed-on: https://chromium-review.googlesource.com/1104377 Reviewed-by: Geoff Lang <geofflang@chromium.org> Reviewed-by: Jamie Madill <jmadill@chromium.org> Commit-Queue: Luc Ferron <lucferron@chromium.org>
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226
//
// Copyright (c) 2014 The ANGLE Project Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
//
// Based on ParticleSystem.c from
// Book: OpenGL(R) ES 2.0 Programming Guide
// Authors: Aaftab Munshi, Dan Ginsburg, Dave Shreiner
// ISBN-10: 0321502795
// ISBN-13: 9780321502797
// Publisher: Addison-Wesley Professional
// URLs: http://safari.informit.com/9780321563835
// http://www.opengles-book.com
#include "SampleApplication.h"
#include "common/vector_utils.h"
#include "shader_utils.h"
#include "random_utils.h"
#include "system_utils.h"
#include "tga_utils.h"
#define _USE_MATH_DEFINES
#include <math.h>
using namespace angle;
class ParticleSystemSample : public SampleApplication
{
public:
ParticleSystemSample(int argc, char **argv) : SampleApplication("ParticleSystem", argc, argv) {}
bool initialize() override
{
const std::string vs =
R"(uniform float u_time;
uniform vec3 u_centerPosition;
attribute float a_lifetime;
attribute vec3 a_startPosition;
attribute vec3 a_endPosition;
varying float v_lifetime;
void main()
{
if (u_time <= a_lifetime)
{
gl_Position.xyz = a_startPosition + (u_time * a_endPosition);
gl_Position.xyz += u_centerPosition;
gl_Position.w = 1.0;
}
else
{
gl_Position = vec4(-1000, -1000, 0, 0);
}
v_lifetime = 1.0 - (u_time / a_lifetime);
v_lifetime = clamp(v_lifetime, 0.0, 1.0);
gl_PointSize = (v_lifetime * v_lifetime) * 40.0;
})";
const std::string fs =
R"(precision mediump float;
uniform vec4 u_color;
varying float v_lifetime;
uniform sampler2D s_texture;
void main()
{
vec4 texColor;
texColor = texture2D(s_texture, gl_PointCoord);
gl_FragColor = vec4(u_color) * texColor;
gl_FragColor.a *= v_lifetime;
})";
mProgram = CompileProgram(vs, fs);
if (!mProgram)
{
return false;
}
// Get the attribute locations
mLifetimeLoc = glGetAttribLocation(mProgram, "a_lifetime");
mStartPositionLoc = glGetAttribLocation(mProgram, "a_startPosition");
mEndPositionLoc = glGetAttribLocation(mProgram, "a_endPosition");
// Get the uniform locations
mTimeLoc = glGetUniformLocation(mProgram, "u_time");
mCenterPositionLoc = glGetUniformLocation(mProgram, "u_centerPosition");
mColorLoc = glGetUniformLocation(mProgram, "u_color");
mSamplerLoc = glGetUniformLocation(mProgram, "s_texture");
glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
// Fill in particle data array
for (size_t i = 0; i < mParticleCount; i++)
{
mParticles[i].lifetime = mRNG.randomFloatBetween(0.0f, 1.0f);
float endAngle = mRNG.randomFloatBetween(0, 2.0f * float(M_PI));
float endRadius = mRNG.randomFloatBetween(0.0f, 2.0f);
mParticles[i].endPosition.x() = sinf(endAngle) * endRadius;
mParticles[i].endPosition.y() = cosf(endAngle) * endRadius;
mParticles[i].endPosition.z() = 0.0f;
float startAngle = mRNG.randomFloatBetween(0, 2.0f * float(M_PI));
float startRadius = mRNG.randomFloatBetween(0.0f, 0.25f);
mParticles[i].startPosition.x() = sinf(startAngle) * startRadius;
mParticles[i].startPosition.y() = cosf(startAngle) * startRadius;
mParticles[i].startPosition.z() = 0.0f;
}
mParticleTime = 1.0f;
std::stringstream smokeStr;
smokeStr << angle::GetExecutableDirectory() << "/smoke.tga";
TGAImage img;
if (!LoadTGAImageFromFile(smokeStr.str(), &img))
{
return false;
}
mTextureID = LoadTextureFromTGAImage(img);
if (!mTextureID)
{
return false;
}
return true;
}
void destroy() override { glDeleteProgram(mProgram); }
void step(float dt, double totalTime) override
{
// Use the program object
glUseProgram(mProgram);
mParticleTime += dt;
if (mParticleTime >= 1.0f)
{
mParticleTime = 0.0f;
// Pick a new start location and color
Vector3 centerPos(mRNG.randomFloatBetween(-0.5f, 0.5f),
mRNG.randomFloatBetween(-0.5f, 0.5f),
mRNG.randomFloatBetween(-0.5f, 0.5f));
glUniform3fv(mCenterPositionLoc, 1, centerPos.data());
// Random color
Vector4 color(mRNG.randomFloatBetween(0.0f, 1.0f), mRNG.randomFloatBetween(0.0f, 1.0f),
mRNG.randomFloatBetween(0.0f, 1.0f), 0.5f);
glUniform4fv(mColorLoc, 1, color.data());
}
// Load uniform time variable
glUniform1f(mTimeLoc, mParticleTime);
}
void draw() override
{
// Set the viewport
glViewport(0, 0, getWindow()->getWidth(), getWindow()->getHeight());
// Clear the color buffer
glClear(GL_COLOR_BUFFER_BIT);
// Use the program object
glUseProgram(mProgram);
// Load the vertex attributes
glVertexAttribPointer(mLifetimeLoc, 1, GL_FLOAT, GL_FALSE, sizeof(Particle), &mParticles[0].lifetime);
glVertexAttribPointer(mEndPositionLoc, 3, GL_FLOAT, GL_FALSE, sizeof(Particle), &mParticles[0].endPosition);
glVertexAttribPointer(mStartPositionLoc, 3, GL_FLOAT, GL_FALSE, sizeof(Particle), &mParticles[0].startPosition);
glEnableVertexAttribArray(mLifetimeLoc);
glEnableVertexAttribArray(mEndPositionLoc);
glEnableVertexAttribArray(mStartPositionLoc);
// Blend particles
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE);
// Bind the texture
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D, mTextureID);
// Set the sampler texture unit to 0
glUniform1i(mSamplerLoc, 0);
glDrawArrays(GL_POINTS, 0, mParticleCount);
}
private:
// Handle to a program object
GLuint mProgram;
// Attribute locations
GLint mLifetimeLoc;
GLint mStartPositionLoc;
GLint mEndPositionLoc;
// Uniform location
GLint mTimeLoc;
GLint mColorLoc;
GLint mCenterPositionLoc;
GLint mSamplerLoc;
// Texture handle
GLuint mTextureID;
// Particle vertex data
struct Particle
{
float lifetime;
Vector3 startPosition;
Vector3 endPosition;
};
static const size_t mParticleCount = 1024;
std::array<Particle, mParticleCount> mParticles;
float mParticleTime;
RNG mRNG;
};
int main(int argc, char **argv)
{
ParticleSystemSample app(argc, argv);
return app.run();
}