123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 |
- using UnityEngine;
- using System.Collections;
- using System.Threading;
- using PPTParticleSystem2D;
- public class PPTGeneratorImage : MonoBehaviour
- {
-
- private PPTParticleSystem physics;
-
-
- public Texture2D image;
-
-
- [Range(0.1f, 100)]
- public float resolution = 6;
-
- public float minimMassValue = 0.4f;
- public float maxMassValue = 0.8f;
-
- public float springConstant=0.02f, damping=0.04f;
-
- [Range(0, 1)]
- public float alphaCutOff = 0;
-
-
-
- public ParticleSystem ParticleSystem;
-
- public PPTForceParticle[] externalForces;
-
- public bool localPosition = false;
-
- public PPTParticleSystem.IntegratorSystem integratorSystem;
-
- public bool isLoaded;
-
- [Range(0, 0.7f)]
- public float somedrag;
-
- public bool springsActivated = true;
-
- public float size = 1;
-
- private int widthSmall, heightSmall, numPixelsSmall;
- private PPTParticle[] particles;
- private PPTSpring[] springs;
- private PPTParticle[] fixedParticles;
- private Color[] colors;
- private float[] forceInstance;
- private bool[] activeInstance;
- private PPTParticle[] instances;
- private PPTAttraction[,] attracts;
- private UnityEngine.ParticleSystem.Particle[] parts;
- private bool m_springsActivated=true;
-
-
- private Thread thread;
- private Mutex mainLoop;
- void OnApplicationQuit()
- {
-
- thread.Abort();
- }
- void Start()
- {
-
- LoadParticles();
- }
-
- public void LoadParticles()
- {
- isLoaded = false;
-
- widthSmall = (int)(image.width / resolution);
- heightSmall = (int)(image.height / resolution);
- numPixelsSmall = 0;
- for (int x = 0; x < widthSmall; x++)
- {
- for (int y = 0; y < heightSmall; y++)
- {
-
- Color c = image.GetPixel((int)(x * resolution), (int)(y * resolution));
- if (c.a > alphaCutOff)
- numPixelsSmall++;
- }
- }
-
- Debug.Log("Generator : PPTParticleSystem - width : " + widthSmall + " - height : " + heightSmall + " - number of particles : " + numPixelsSmall);
-
- colors = new Color[numPixelsSmall];
-
- physics = new PPTParticleSystem(0f, somedrag);
-
- physics.setIntegrator(integratorSystem);
-
-
- particles = new PPTParticle[numPixelsSmall];
-
- fixedParticles = new PPTParticle[numPixelsSmall];
-
- springs = new PPTSpring[numPixelsSmall];
-
- attracts = new PPTAttraction[externalForces.Length, numPixelsSmall];
-
- instances = new PPTParticle[externalForces.Length];
- forceInstance = new float[externalForces.Length];
- activeInstance = new bool[externalForces.Length];
-
- Vector3 v = new Vector3(0, 0, 0);
- for (int i = 0; i < externalForces.Length; i++)
- {
- if (localPosition)
- v = externalForces[i].transform.localPosition;
- else
- v = externalForces[i].transform.position;
-
-
- instances[i] = physics.makeParticle(1, v.x, v.y, v.z);
- instances[i].makeFixed();
- forceInstance[i] = externalForces[i].forceParticle;
- activeInstance[i] = externalForces[i].gameObject.activeSelf;
- }
-
- int a = 0;
- for (int x = 0; x < widthSmall; x++)
- {
- for (int y = 0; y < heightSmall; y++)
- {
-
- Color c = image.GetPixel((int)(x * resolution), (int)(y * resolution));
-
- if (c.a > alphaCutOff)
- {
-
- colors[a] = image.GetPixel((int)(x * resolution), (int)(y * resolution));
-
- particles[a] = physics.makeParticle(
- Random.Range(minimMassValue, maxMassValue),
- (x * resolution - widthSmall * resolution / 2)/(60/size),
- (y * resolution - heightSmall * resolution / 2)/(60/size), 0);
-
- fixedParticles[a] = physics.makeParticle(
- Random.Range(minimMassValue, maxMassValue),
- (x * resolution - widthSmall * resolution / 2)/(60/size),
- (y * resolution - heightSmall * resolution / 2)/(60/size), 0);
-
- fixedParticles[a].makeFixed();
- int i = 0;
-
- foreach (PPTParticle p in instances)
- {
-
- attracts[i, a] = physics.makeAttraction(particles[a], p, forceInstance[i], 0.1f);
- if (!activeInstance[i])
-
- attracts[i, a].turnOff();
- i++;
- }
-
- springs[a] = physics.makeSpring(particles[a], fixedParticles[a], springConstant, damping, 0);
- a++;
- }
- }
- }
-
- isLoaded = true;
-
- parts = new ParticleSystem.Particle[numPixelsSmall];
- mainLoop = new Mutex(true);
- thread = new Thread(runPhysics);
-
- thread.Start();
-
-
- if (ParticleSystem != null)
- {
- float particleSize = ParticleSystem.startSize;
- int i = numPixelsSmall;
- while (--i > -1)
- {
- ParticleSystem.Particle particle = new ParticleSystem.Particle();
- particle.position = new Vector2(particles[i].position.x,particles[i].position.y);
- particle.startLifetime = float.MaxValue;
- particle.remainingLifetime = float.MaxValue;
- particle.size = particleSize;
- particle.color = colors[i];
- parts[i] = particle;
- }
- ParticleSystem.SetParticles(parts, parts.Length);
- ParticleSystem.Play();
- }
- else {
- Debug.LogError("Particle system not setted! Please attach one Unity particle system to the PPTParticleSystem.");
- }
- }
-
- void Update()
- {
-
- updateExternalForces();
-
- if (physics != null)
- updateParticleSystem();
-
- updateSpringSystem();
- mainLoop.ReleaseMutex();
- mainLoop.WaitOne();
- }
- private void updateSpringSystem()
- {
-
- if (m_springsActivated != springsActivated)
- {
- m_springsActivated = springsActivated;
- foreach (PPTSpring s in springs)
- if (m_springsActivated)
- s.turnOn();
- else
- s.turnOff();
- }
- }
- private void updateExternalForces()
- {
- Vector3 v = new Vector3(0, 0, 0);
- for (int i = 0; i < externalForces.Length; i++)
- {
- if (localPosition)
- v = externalForces[i].transform.localPosition;
- else
- v = externalForces[i].transform.position;
-
- instances[i].position.x = v.x;
- instances[i].position.y = v.y;
- if (forceInstance[i] != externalForces[i].forceParticle || activeInstance[i] != externalForces[i].gameObject.activeSelf)
- {
- forceInstance[i] = externalForces[i].forceParticle;
- activeInstance[i] = externalForces[i].gameObject.activeSelf;
- if (activeInstance[i])
- for (int a = 0; a < numPixelsSmall; a++)
- {
-
- attracts[i, a].turnOn();
- attracts[i, a].setStrength(forceInstance[i]);
- }
- else
- for (int a = 0; a < numPixelsSmall; a++)
-
- attracts[i, a].turnOff();
- }
- }
- }
- public void runPhysics()
- {
- while (true)
- {
-
- Thread.Sleep(0);
- physics.tick();
- mainLoop.WaitOne();
- mainLoop.ReleaseMutex();
- }
- }
- public void updateParticleSystem()
- {
-
- ParticleSystem.GetParticles(parts);
-
- int i = Mathf.Min(numPixelsSmall, particles.Length);
- while (--i > -1)
- {
- parts[i].position = new Vector2(particles[i].position.x, particles[i].position.y);
- parts[i].remainingLifetime = float.MaxValue;
- }
-
- ParticleSystem.SetParticles(parts, particles.Length);
-
- }
- }
|