├── .npmignore
├── examples
├── capture.jpg
├── grass_diffuse.png
└── basic.html
├── src
├── constants.js
├── three-particle-fire.js
├── install.js
├── geometry.js
├── material.js
└── texture.js
├── .gitignore
├── .babelrc
├── package.json
├── rollup.config.js
├── readme.md
└── dist
├── three-particle-fire.min.js
├── three-particle-fire.module.js
└── three-particle-fire.js
/.npmignore:
--------------------------------------------------------------------------------
1 | examples/*
2 |
--------------------------------------------------------------------------------
/examples/capture.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/yomotsu/three-particle-fire/HEAD/examples/capture.jpg
--------------------------------------------------------------------------------
/examples/grass_diffuse.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/yomotsu/three-particle-fire/HEAD/examples/grass_diffuse.png
--------------------------------------------------------------------------------
/src/constants.js:
--------------------------------------------------------------------------------
1 | export const SPRITE_ROW_LENGTH = 4;
2 | export const ONE_SPRITE_ROW_LENGTH = 1 / SPRITE_ROW_LENGTH;
3 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | *.DS_Store
2 | Thumbs.db
3 |
4 | logs
5 | *.log
6 | npm-debug.log*
7 |
8 | *.map
9 |
10 | node_modules
11 |
--------------------------------------------------------------------------------
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "presets": [
3 | [ "env", {
4 | "targets": {
5 | "browsers": [
6 | "last 2 versions",
7 | "ie >= 11"
8 | ]
9 | },
10 | "loose": true,
11 | "modules": false
12 | } ]
13 | ]
14 | }
15 |
--------------------------------------------------------------------------------
/src/three-particle-fire.js:
--------------------------------------------------------------------------------
1 | import makeGeometryClass from './geometry.js';
2 | import makeMaterialClass from './material.js';
3 | import { install, onInstall } from './install.js';
4 |
5 | const particleFire = {
6 | // Geometry,
7 | // Material,
8 | install
9 | };
10 |
11 | onInstall( 'THREE', () => {
12 |
13 | particleFire.Geometry = makeGeometryClass();
14 | particleFire.Material = makeMaterialClass();
15 |
16 | } );
17 |
18 | export default particleFire;
19 |
--------------------------------------------------------------------------------
/src/install.js:
--------------------------------------------------------------------------------
1 | const libs = {};
2 | const callbacks = {};
3 |
4 | export function install( _libs ) {
5 |
6 | libs.THREE = _libs.THREE;
7 |
8 | for ( let i = 0, l = callbacks[ 'THREE' ].length; i < l; i ++ ) {
9 |
10 | callbacks[ 'THREE' ][ i ]();
11 |
12 | }
13 |
14 | delete callbacks[ 'THREE' ];
15 |
16 | }
17 |
18 | export function getInstalled( key ) {
19 |
20 | return libs[ key ];
21 |
22 | }
23 |
24 | export function onInstall( key, callback ) {
25 |
26 | if ( ! callbacks[ key ] ) callbacks[ key ] = [];
27 |
28 | callbacks[ key ].push( callback );
29 |
30 | }
31 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "three-particle-fire",
3 | "version": "0.1.0",
4 | "author": "Yomotsu",
5 | "license": "MIT",
6 | "main": "dist/three-particle-fire.js",
7 | "repository": "yomotsu/three-particle-fire",
8 | "jsnext:main": "dist/three-particle-fire.module.js",
9 | "module": "dist/three-particle-fire.module.js",
10 | "devDependencies": {
11 | "babel-core": "^6.26.3",
12 | "babel-preset-env": "1.7.0",
13 | "rollup": "^2.77.3",
14 | "rollup-plugin-babel": "3.0.2",
15 | "rollup-plugin-commonjs": "8.2.6",
16 | "rollup-plugin-node-resolve": "^3.0.0",
17 | "rollup-watch": "^4.3.1",
18 | "uglify-js": "^3.16.3"
19 | },
20 | "scripts": {
21 | "dev": "npm run watch",
22 | "watch": "rollup --config --watch",
23 | "build": "rollup --config",
24 | "release": "rollup -c && uglifyjs dist/three-particle-fire.js -cm --preamble \"/*!\n * three-particle-fire\n * (c) 2017 @yomotsu\n * Released under the MIT License.\n */\" > dist/three-particle-fire.min.js"
25 | }
26 | }
27 |
--------------------------------------------------------------------------------
/rollup.config.js:
--------------------------------------------------------------------------------
1 | import nodeResolve from 'rollup-plugin-node-resolve';
2 | import commonjs from 'rollup-plugin-commonjs';
3 | import babel from 'rollup-plugin-babel';
4 | import pkg from './package.json';
5 |
6 | const license = `/*!
7 | * ${ pkg.name }
8 | * https://github.com/${ pkg.repository }
9 | * (c) 2017 @yomotsu
10 | * Released under the MIT License.
11 | */`
12 |
13 | export default {
14 | input: 'src/three-particle-fire.js',
15 | output: [
16 | {
17 | format: 'umd',
18 | name: 'particleFire',
19 | file: 'dist/three-particle-fire.js',
20 | banner: license
21 | },
22 | {
23 | format: 'es',
24 | file: 'dist/three-particle-fire.module.js',
25 | banner: license
26 | }
27 | ],
28 | indent: '\t',
29 | sourceMap: false,
30 | plugins: [
31 | nodeResolve( {
32 | jsnext: true,
33 | browser: true,
34 | preferBuiltins: false
35 | } ),
36 | commonjs( {
37 | include: [ 'node_modules/**' ],
38 | exclude: [],
39 | sourceMap: true,
40 | } ),
41 | babel( {
42 | exclude: 'node_modules/**',
43 | presets: [
44 | [ 'env', {
45 | targets: {
46 | browsers: [ 'last 2 versions' ]
47 | },
48 | loose: true,
49 | modules: false
50 | } ]
51 | ]
52 | } )
53 | ]
54 | };
55 |
--------------------------------------------------------------------------------
/src/geometry.js:
--------------------------------------------------------------------------------
1 | import { ONE_SPRITE_ROW_LENGTH } from './constants.js';
2 | import { getInstalled } from './install.js';
3 |
4 | export default function makeGeometryClass() {
5 |
6 | const THREE = getInstalled( 'THREE' );
7 |
8 | return class Geometry {
9 |
10 | constructor( radius, height, particleCount ) {
11 |
12 | const geometry = new THREE.BufferGeometry();
13 |
14 | const halfHeight = height * 0.5;
15 | const position = new Float32Array( particleCount * 3 );
16 | const random = new Float32Array( particleCount );
17 | const sprite = new Float32Array( particleCount );
18 |
19 | for ( let i = 0; i < particleCount; i ++ ) {
20 |
21 | const r = Math.sqrt( Math.random() ) * radius;
22 | const angle = Math.random() * 2 * Math.PI;
23 | position[ i * 3 + 0 ] = Math.cos( angle ) * r;
24 | position[ i * 3 + 1 ] = ( radius - r ) / radius * halfHeight + halfHeight;
25 | position[ i * 3 + 2 ] = Math.sin( angle ) * r;
26 | sprite[ i ] = ONE_SPRITE_ROW_LENGTH * ( ( Math.random() * 4 )|0 );
27 | random[ i ] = Math.random();
28 |
29 | if ( i === 0 ) {
30 |
31 | // to avoid going out of Frustum
32 | position[ i * 3 + 0 ] = 0;
33 | position[ i * 3 + 1 ] = 0;
34 | position[ i * 3 + 2 ] = 0;
35 |
36 | }
37 |
38 | }
39 |
40 | geometry.setAttribute( 'position', new THREE.BufferAttribute( position, 3 ) );
41 | geometry.setAttribute( 'random', new THREE.BufferAttribute( random, 1 ) );
42 | geometry.setAttribute( 'sprite', new THREE.BufferAttribute( sprite, 1 ) );
43 | return geometry;
44 |
45 | }
46 |
47 | }
48 |
49 | }
50 |
--------------------------------------------------------------------------------
/readme.md:
--------------------------------------------------------------------------------
1 | # three-particle-fire
2 |
3 | Fire mesh object for three.js
4 |
5 | [](https://www.npmjs.com/package/three-particle-fire)
6 | 
7 | [](https://david-dm.org/yomotsu/three-particle-fire)
8 |
9 | 
10 |
11 | [Demos can be seen here](https://yomotsu.github.io/three-particle-fire/examples/basic.html).
12 |
13 | ## Usage
14 |
15 | 1. Import your three.js and then prepare three-particle-fire to use.
16 | ```javascript
17 | import * as THREE from 'three';
18 | import particleFire from 'three-particle-fire';
19 |
20 | particleFire.install( { THREE: THREE } );
21 | ```
22 |
23 | 2. Now ready to use. particleFire will provide geometry and material for `THREE.Points` class.
24 | ```javascript
25 | var fireRadius = 0.5;
26 | var fireHeight = 3;
27 | var particleCount = 800;
28 | var height = window.innerHeight;
29 |
30 | var geometry0 = new particleFire.Geometry( fireRadius, fireHeight, particleCount );
31 | var material0 = new particleFire.Material( { color: 0xff2200 } );
32 | material0.setPerspective( camera.fov, height );
33 | var particleFireMesh0 = new THREE.Points( geometry0, material0 );
34 | scene.add( particleFireMesh0 );
35 | ```
36 |
37 | 3. Update on tick in your render loop.
38 | ```javascript
39 | function update () {
40 |
41 | var delta = clock.getDelta();
42 |
43 | requestAnimationFrame( update );
44 |
45 | particleFireMesh0.material.update( delta );
46 | renderer.render( scene, camera );
47 |
48 | }
49 | ```
50 |
51 | 4. Sync on onresize event
52 |
53 | ```javascript
54 | window.addEventListener( 'resize', function () {
55 |
56 | var width = window.innerWidth;
57 | var height = window.innerHeight;
58 |
59 | camera.aspect = width / height;
60 | camera.updateProjectionMatrix();
61 | renderer.setSize( width, height );
62 |
63 | particleFireMesh0.material.setPerspective( camera.fov, height );
64 |
65 | } );
66 | ```
67 |
--------------------------------------------------------------------------------
/src/material.js:
--------------------------------------------------------------------------------
1 | import { ONE_SPRITE_ROW_LENGTH } from './constants.js';
2 | import { getInstalled } from './install.js';
3 | import { getTexture } from './texture.js';
4 |
5 | export default function makeMaterialClass() {
6 |
7 | const THREE = getInstalled( 'THREE' );
8 |
9 | return class Material {
10 |
11 | constructor( parameters ) {
12 |
13 | const uniforms = {
14 | color : { value: null },
15 | size : { value: 0.0 },
16 | map : { value: getTexture() },
17 | time : { value: 0.0 },
18 | heightOfNearPlane: { value: 0.0 }
19 | };
20 |
21 | const material = new THREE.ShaderMaterial({
22 |
23 | uniforms : uniforms,
24 |
25 | vertexShader : [
26 | 'attribute float random;',
27 | 'attribute float sprite;',
28 | 'uniform float time;',
29 | 'uniform float size;',
30 | 'uniform float heightOfNearPlane;',
31 |
32 | 'varying float vSprite;',
33 | 'varying float vOpacity;',
34 |
35 | 'float PI = 3.14;',
36 |
37 | 'float quadraticIn( float t ) {',
38 |
39 | 'float tt = t * t;',
40 | 'return tt * tt;',
41 |
42 | '}',
43 |
44 | 'void main() {',
45 |
46 | 'float progress = fract( time + ( 2.0 * random - 1.0 ) );',
47 | 'float progressNeg = 1.0 - progress;',
48 | 'float ease = quadraticIn( progress );',
49 | 'float influence = sin( PI * ease );',
50 |
51 | 'vec3 newPosition = position * vec3( 1.0, ease, 1.0 );',
52 | 'gl_Position = projectionMatrix * modelViewMatrix * vec4( newPosition, 1.0 );',
53 | 'gl_PointSize = ( heightOfNearPlane * size ) / gl_Position.w;',
54 |
55 | 'vOpacity = min( influence * 4.0, 1.0 ) * progressNeg;',
56 | 'vSprite = sprite;',
57 |
58 | '}'
59 | ].join( '\n' ),
60 |
61 | fragmentShader: [
62 | 'uniform vec3 color;',
63 | 'uniform sampler2D map;',
64 |
65 | 'varying float vSprite;',
66 | 'varying float vOpacity;',
67 |
68 | 'void main() {',
69 |
70 | 'vec2 texCoord = vec2(',
71 | 'gl_PointCoord.x * ' + ONE_SPRITE_ROW_LENGTH + ' + vSprite,',
72 | 'gl_PointCoord.y',
73 | ');',
74 |
75 | 'gl_FragColor = vec4( texture2D( map, texCoord ).xyz * color * vOpacity, 1.0 );',
76 |
77 | '}'
78 | ].join( '\n' ),
79 |
80 | blending : THREE.AdditiveBlending,
81 | depthTest : true,
82 | depthWrite : false,
83 | transparent: true,
84 | // fog : true
85 |
86 | } );
87 |
88 | material.color = new THREE.Color( 0xff2200 );
89 | material.size = 0.4;
90 |
91 | if ( parameters !== undefined ) {
92 | material.setValues( parameters );
93 | }
94 |
95 | material.uniforms.color.value = material.color;
96 | material.uniforms.size.value = material.size;
97 |
98 | material.update = function( delta ) {
99 |
100 | material.uniforms.time.value = ( material.uniforms.time.value + delta ) % 1;
101 |
102 | }
103 |
104 | material.setPerspective = function( fov, height ) {
105 |
106 | material.uniforms.heightOfNearPlane.value = Math.abs( height / ( 2 * Math.tan( THREE.MathUtils.degToRad( fov * 0.5 ) ) ) );
107 |
108 | }
109 |
110 | return material;
111 |
112 | }
113 |
114 | }
115 |
116 | }
117 |
--------------------------------------------------------------------------------
/examples/basic.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 | =^.^=
6 |
10 |
11 |
12 |
13 |
14 |
15 |
136 |
137 |
138 |
139 |
--------------------------------------------------------------------------------
/src/texture.js:
--------------------------------------------------------------------------------
1 | import { getInstalled } from './install.js';
2 |
3 | let texture;
4 |
5 | export function getTexture() {
6 |
7 | if ( !! texture ) return texture;
8 |
9 | const THREE = getInstalled( 'THREE' );
10 | const image = new Image();
11 |
12 | texture = new THREE.Texture();
13 | texture.wrapS = texture.wrapT = THREE.RepeatWrapping;
14 | texture.image = image;
15 |
16 | image.onload = () => {
17 |
18 | texture.needsUpdate = true;
19 |
20 | };
21 |
22 | image.src = '';
23 |
24 | return texture;
25 |
26 | }
27 |
--------------------------------------------------------------------------------
/dist/three-particle-fire.min.js:
--------------------------------------------------------------------------------
1 | !function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):t.particleFire=e()}(this,function(){"use strict";function t(t){return A[t]}function e(){var e=t("THREE"),A=function(A){var o={color:{type:"c",value:null},size:{type:"f",value:0},texture:{type:"t",value:function(){if(n)return n;var e=t("THREE"),r=new Image;return n=new e.Texture,n.wrapS=n.wrapT=e.RepeatWrapping,n.image=r,r.onload=function(){n.needsUpdate=!0},r.src="",n}()},time:{type:"f",value:0},heightOfNearPlane:{type:"f",value:0}};e.ShaderMaterial.call(this,{uniforms:o,vertexShader:["attribute float randam;","attribute float sprite;","uniform float time;","uniform float size;","uniform float heightOfNearPlane;","varying float vSprite;","varying float vOpacity;","float PI = 3.14;","float quadraticIn( float t ) {","float tt = t * t;","return tt * tt;","}","void main() {","float progress = fract( time + ( 2.0 * randam - 1.0 ) );","float progressNeg = 1.0 - progress;","float ease = quadraticIn( progress );","float influence = sin( PI * ease );","vec3 newPosition = position * vec3( 1.0, ease, 1.0 );","gl_Position = projectionMatrix * modelViewMatrix * vec4( newPosition, 1.0 );","gl_PointSize = ( heightOfNearPlane * size ) / gl_Position.w;","vOpacity = min( influence * 4.0, 1.0 ) * progressNeg;","vSprite = sprite;","}"].join("\n"),fragmentShader:["uniform vec3 color;","uniform sampler2D texture;","varying float vSprite;","varying float vOpacity;","void main() {","vec2 texCoord = vec2(","gl_PointCoord.x * "+r+" + vSprite,","gl_PointCoord.y",");","gl_FragColor = vec4( texture2D( texture, vec2( texCoord ) ).xyz * color * vOpacity, 1.0 );","}"].join("\n"),blending:e.AdditiveBlending,depthTest:!0,depthWrite:!1,transparent:!0}),this.color=new e.Color(16720384),this.size=.4,this.setValues(A),this.uniforms.color.value=this.color,this.uniforms.size.value=this.size};return A.prototype=Object.create(e.ShaderMaterial.prototype),A.prototype.update=function(t){this.uniforms.time.value=(this.uniforms.time.value+t)%1},A.prototype.setPerspective=function(t,r){this.uniforms.heightOfNearPlane.value=Math.abs(r/(2*Math.tan(e.Math.degToRad(.5*t))))},A}var r=.25,A={},o={},n=void 0,i={install:function(t){A.THREE=t.THREE;for(var e=0,r=o.THREE.length;e