├── .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 | [![Latest NPM release](https://img.shields.io/npm/v/three-particle-fire.svg)](https://www.npmjs.com/package/three-particle-fire) 6 | ![MIT License](https://img.shields.io/npm/l/three-particle-fire.svg) 7 | [![dependencies Status](https://david-dm.org/yomotsu/three-particle-fire/status.svg)](https://david-dm.org/yomotsu/three-particle-fire) 8 | 9 | ![](https://yomotsu.github.io/three-particle-fire/examples/capture.jpg) 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