Différences
Ci-dessous, les différences entre deux révisions de la page.
Les deux révisions précédentes Révision précédente Prochaine révision | Révision précédente | ||
recherche:residence_infra:start [2024/11/17 11:37] gweltaz [Quaternion] |
recherche:residence_infra:start [2024/11/22 18:03] (Version actuelle) exsitu [Infra-Documentation (ExSitu)] |
||
---|---|---|---|
Ligne 1: | Ligne 1: | ||
======INFRA====== | ======INFRA====== | ||
+ | |||
+ | ===== Infra-Documentation (ExSitu) ===== | ||
+ | |||
+ | Test de l'outil Multitude fraîchement développer par Laurent et permettant de collecter du son directement depuis une appli web, de l'archiver et de le rediffuser. | ||
+ | |||
+ | Le résultat sur 4 jours de résidence : https://multitude.exsitu.xyz/v/visu/1931309934af589f4 | ||
+ | |||
+ | La version map GPS : https://multitude.exsitu.xyz/v/map/1931309934af589f4 | ||
+ | |||
+ | {{:recherche:residence_infra:inframultitude.png?nolink&400|}} | ||
+ | |||
+ | Démonstration de la diffusion sur 7 enceintes : | ||
+ | |||
+ | {{:recherche:residence_infra:infraenceintesx7.png?nolink&400|}} | ||
+ | |||
+ | Gitlab : https://gitlab.com/losylam/multitude | ||
===== Laurent ===== | ===== Laurent ===== | ||
Ligne 24: | Ligne 40: | ||
- | Profiter de ces beaux jours d'automne pour se chauffer au néons et ressortir la petite [[bobineuse https://ressources.labomedia.org/bobineuse?s[]=bobineuse]] pour créer un mirco | + | Profiter de ces beaux jours d'automne pour se chauffer au néons et ressortir la petite [[https://ressources.labomedia.org/bobineuse?s|bobineuse]] pour créer un mirco |
====la base infrastructurelle==== | ====la base infrastructurelle==== | ||
Ligne 62: | Ligne 78: | ||
===== Quaternion ===== | ===== Quaternion ===== | ||
+ | |||
+ | **Exploration des quaternions et de la modélisation procedurale avec Processing** | ||
+ | |||
+ | Une vidéo qui m'a bien aidée à comprendre le ce truc : https://www.youtube.com/watch?v=bKd2lPjl92c | ||
{{:recherche:residence_infra:capture_d_ecran_du_2024-11-10_23-26-27.png?nolink&400|}} | {{:recherche:residence_infra:capture_d_ecran_du_2024-11-10_23-26-27.png?nolink&400|}} | ||
Ligne 69: | Ligne 89: | ||
{{:recherche:residence_infra:capture_d_ecran_du_2024-11-11_21-43-18.png?nolink&400|}} | {{:recherche:residence_infra:capture_d_ecran_du_2024-11-11_21-43-18.png?nolink&400|}} | ||
{{:recherche:residence_infra:capture_d_ecran_du_2024-11-11_23-24-57.png?nolink&400|}} | {{:recherche:residence_infra:capture_d_ecran_du_2024-11-11_23-24-57.png?nolink&400|}} | ||
+ | |||
+ | Les sketchs Processing ci-dessous dépendent de façon importante sur deux librairies non officielles: | ||
+ | |||
+ | * **QueasyCam** (modifié), pour la navigation dans l'espace en 3D -> https://github.com/gweltou/queasycam/tree/master/distribution/queasycam-6/download/ | ||
+ | * **LibAvatar**, pour l'intégration des classes de LibGDX dans Processing -> https://github.com/gweltou/Processing-libAvatar/blob/2.0/distribution/libAvatar-3/download/libAvatar.zip | ||
+ | |||
+ | Vous pouvez naviguer dans l'espace 3D avec les touches <key>z</key><key>q</key><key>s</key><key>d</key><key>a</key> et <key>e</key>. Affichage en mode filaire avec <key>w</key>. | ||
<accordion><panel title="iteration1.pde"> | <accordion><panel title="iteration1.pde"> | ||
Ligne 293: | Ligne 320: | ||
for (Segment child : children) | for (Segment child : children) | ||
child.update(); | child.update(); | ||
+ | } | ||
+ | } | ||
+ | </code> | ||
+ | </panel></accordion> | ||
+ | |||
+ | <accordion><panel title="iteration2.pde"> | ||
+ | <code java> | ||
+ | import queasycam.*; | ||
+ | import com.badlogic.gdx.math.*; | ||
+ | import nervoussystem.obj.*; | ||
+ | import processing.pdf.*; | ||
+ | |||
+ | static final Vector3 xAxis = new Vector3(1.0, 0.0, 0.0); | ||
+ | static final Vector3 yAxis = new Vector3(0.0, 1.0, 0.0); | ||
+ | static final Vector3 zAxis = new Vector3(0.0, 0.0, 1.0); | ||
+ | |||
+ | |||
+ | QueasyCam cam; | ||
+ | |||
+ | boolean wireframe = false; | ||
+ | boolean record = false; | ||
+ | |||
+ | ArrayList<Drawable> objects = new ArrayList(); | ||
+ | |||
+ | |||
+ | public void setup() { | ||
+ | fullScreen(P3D); | ||
+ | //size(800, 600, P3D); | ||
+ | |||
+ | cam = new QueasyCam(this, 1, 9999); | ||
+ | cam.key_forward = 'z'; | ||
+ | cam.key_left = 'q'; | ||
+ | cam.key_backward = 's'; | ||
+ | cam.key_right = 'd'; | ||
+ | cam.key_up = 'e'; | ||
+ | cam.key_down = 'a'; | ||
+ | | ||
+ | Pipe pipe = new Pipe( | ||
+ | new Vector3(0, 0, 0), new Quaternion().setEulerAngles(10, 0, 40), | ||
+ | 10, 32); | ||
+ | objects.add(pipe); | ||
+ | float size = 20; | ||
+ | for (int i=0; i<8; i++) { | ||
+ | pipe.changeSize(size); | ||
+ | size += 10; | ||
+ | pipe.addBend(90, 0.0, 45.0); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | background(255); | ||
+ | lights(); | ||
+ | |||
+ | strokeWeight(1); | ||
+ | stroke(255, 0, 0); | ||
+ | line(0, 0, 0, 100, 0, 0); | ||
+ | stroke(0, 255, 0); | ||
+ | line(0, 0, 0, 0, 100, 0); | ||
+ | stroke(0, 0, 255); | ||
+ | line(0, 0, 0, 0, 0, 100); | ||
+ | |||
+ | if (wireframe) | ||
+ | stroke(0); | ||
+ | else | ||
+ | noStroke(); | ||
+ | strokeWeight(2); | ||
+ | | ||
+ | if (record) { | ||
+ | beginRaw(PDF, "output.pdf"); | ||
+ | noFill(); | ||
+ | } | ||
+ | |||
+ | for (Drawable o : objects) | ||
+ | o.draw(); | ||
+ | | ||
+ | if (record) { | ||
+ | record = false; | ||
+ | endRaw(); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | void keyPressed() { | ||
+ | if (key == 'w') | ||
+ | wireframe = !wireframe; | ||
+ | | ||
+ | if (key == 'p') | ||
+ | record = true; | ||
+ | } | ||
+ | |||
+ | |||
+ | |||
+ | public interface Drawable { | ||
+ | public void draw(); | ||
+ | public void update(); | ||
+ | } | ||
+ | |||
+ | public interface Chainable extends Drawable { | ||
+ | public Vector3 getOutPos(); | ||
+ | public Quaternion getOutOrientation(); | ||
+ | } | ||
+ | |||
+ | |||
+ | /***************************** | ||
+ | ** class Pipe ** | ||
+ | *****************************/ | ||
+ | |||
+ | public class Pipe implements Drawable { | ||
+ | private Vector3 pos; | ||
+ | private Quaternion orientation; | ||
+ | private float radius; | ||
+ | private float nextRadius; | ||
+ | private int faces; | ||
+ | private ArrayList<Chainable> objects = new ArrayList(); | ||
+ | private Vector3 currentPos; | ||
+ | private Quaternion currentOrientation; | ||
+ | |||
+ | public Pipe(Vector3 pos, Quaternion orientation, float radius, int faces) { | ||
+ | this.pos = pos; | ||
+ | this.orientation = orientation; | ||
+ | this.radius = radius; | ||
+ | this.nextRadius = radius; | ||
+ | this.faces = faces; | ||
+ | currentPos = this.pos.cpy(); | ||
+ | currentOrientation = orientation.cpy(); | ||
+ | } | ||
+ | |||
+ | public void changeSize(float next) { | ||
+ | this.nextRadius = next; | ||
+ | } | ||
+ | | ||
+ | private void add(Chainable object) { | ||
+ | objects.add(object); | ||
+ | currentPos = object.getOutPos(); | ||
+ | currentOrientation = object.getOutOrientation(); | ||
+ | radius = nextRadius; | ||
+ | } | ||
+ | |||
+ | public void addTube(float len) { | ||
+ | addTube(len, 0.0); | ||
+ | } | ||
+ | | ||
+ | public void addTube(float len, float twist) { | ||
+ | Tube tube = new Tube( | ||
+ | currentPos, currentOrientation, | ||
+ | len, radius, nextRadius, twist, 1, faces | ||
+ | ); | ||
+ | add(tube); | ||
+ | } | ||
+ | | ||
+ | public void addBend(float angleBendDeg) { | ||
+ | addBend(angleBendDeg, 0.0, 0.0); | ||
+ | } | ||
+ | | ||
+ | public void addBend(float angleBendDeg, float angleRotDeg) { | ||
+ | addBend(angleBendDeg, angleRotDeg, 0.0); | ||
+ | } | ||
+ | |||
+ | public void addBend(float angleBendDeg, float angleRotDeg, float twist) { | ||
+ | int rings = 8; | ||
+ | float radInt = 50; | ||
+ | currentOrientation.mul(new Quaternion().setEulerAngles(0, angleRotDeg, 0)); | ||
+ | Bend bend = new Bend( | ||
+ | currentPos, currentOrientation, | ||
+ | radius, nextRadius, | ||
+ | radians(angleBendDeg), radInt, twist, | ||
+ | rings, faces | ||
+ | ); | ||
+ | add(bend); | ||
+ | } | ||
+ | |||
+ | public void update() { | ||
+ | for (Chainable o : objects) | ||
+ | o.update(); | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | for (Chainable o : objects) { | ||
+ | o.draw(); | ||
+ | } | ||
+ | |||
+ | Vector3 head = currentOrientation.transform(new Vector3(32, 0, 0)); | ||
+ | head.add(currentPos); | ||
+ | line(currentPos.x, currentPos.y, currentPos.z, head.x, head.y, head.z); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | /***************************** | ||
+ | ** class Tube ** | ||
+ | *****************************/ | ||
+ | |||
+ | public class Tube implements Drawable, Chainable { | ||
+ | private Vector3 pos; | ||
+ | private Quaternion orientation; | ||
+ | private float len; | ||
+ | private float radiusIn; | ||
+ | private float radiusOut; | ||
+ | private float twist; | ||
+ | private int rings; | ||
+ | private int faces; | ||
+ | private Vector3[] points; | ||
+ | private Vector3 tmpVec = new Vector3(); | ||
+ | private Vector3 outPos = new Vector3(); | ||
+ | private Quaternion outOrientation = new Quaternion(); | ||
+ | |||
+ | public Tube( | ||
+ | Vector3 pos, | ||
+ | Quaternion orientation, | ||
+ | float len, | ||
+ | float radiusIn, | ||
+ | float radiusOut, | ||
+ | float twist, // In degrees | ||
+ | int rings, int faces | ||
+ | ) { | ||
+ | this.pos = pos.cpy(); | ||
+ | this.orientation = orientation.cpy(); | ||
+ | this.len = len; | ||
+ | this.radiusIn = radiusIn; | ||
+ | this.radiusOut = radiusOut; | ||
+ | this.twist = radians(twist); | ||
+ | this.rings = rings; | ||
+ | this.faces = faces; | ||
+ | points = new Vector3[faces * (rings+1)]; | ||
+ | update(); | ||
+ | } | ||
+ | | ||
+ | Vector3 getOutPos() { | ||
+ | return outPos.cpy(); | ||
+ | } | ||
+ | | ||
+ | Quaternion getOutOrientation() { | ||
+ | return outOrientation.cpy(); | ||
+ | } | ||
+ | | ||
+ | public void update() { | ||
+ | Quaternion rot = new Quaternion(); | ||
+ | int pointIdx = 0; | ||
+ | for (int s=0; s<rings+1; s++) { | ||
+ | float angle = s * twist/rings; | ||
+ | for (int i=0; i<faces; i++) { | ||
+ | tmpVec.set(s * len/rings, map(s, 0, rings, radiusIn, radiusOut), 0.0); | ||
+ | rot.setFromAxisRad(xAxis, angle).mulLeft(orientation); | ||
+ | rot.transform(tmpVec); | ||
+ | points[pointIdx++] = tmpVec.cpy().add(pos); | ||
+ | angle += TWO_PI / faces; | ||
+ | } | ||
+ | } | ||
+ | outOrientation.set(orientation); | ||
+ | outOrientation.mul(new Quaternion().setFromAxisRad(xAxis, twist)); | ||
+ | outPos.set(len, 0.0, 0.0); | ||
+ | outOrientation.transform(outPos); | ||
+ | outPos.add(pos); | ||
+ | | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | for (int s=0; s<rings; s++) { | ||
+ | beginShape(QUAD_STRIP); | ||
+ | for (int i=0; i<faces+1; i++) { | ||
+ | tmpVec.set(points[s * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | tmpVec.set(points[(s+1) * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | } | ||
+ | endShape(); | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | /***************************** | ||
+ | ** class Bend ** | ||
+ | *****************************/ | ||
+ | |||
+ | public class Bend implements Drawable, Chainable { | ||
+ | private Vector3 pos; | ||
+ | private Quaternion orientation; | ||
+ | private float radiusIn; | ||
+ | private float radiusOut; | ||
+ | private float bendAngle; | ||
+ | private float radInt; | ||
+ | private float twist; | ||
+ | private int rings; | ||
+ | private int faces; | ||
+ | private Vector3[] points; | ||
+ | private Vector3 tmpVec = new Vector3(); | ||
+ | private Vector3 outPos = new Vector3(); | ||
+ | private Quaternion outOrientation = new Quaternion(); | ||
+ | |||
+ | public Bend( | ||
+ | Vector3 pos, | ||
+ | Quaternion orientation, | ||
+ | float radiusIn, | ||
+ | float radiusOut, | ||
+ | float bendAngle, // In radians | ||
+ | float radInt, // Internal radius | ||
+ | float twist, | ||
+ | int rings, | ||
+ | int faces | ||
+ | ) { | ||
+ | this.pos = pos.cpy(); | ||
+ | this.orientation = orientation.cpy(); | ||
+ | this.radiusIn = radiusIn; | ||
+ | this.radiusOut = radiusOut; | ||
+ | this.bendAngle = bendAngle; | ||
+ | this.radInt = radInt; | ||
+ | this.twist = radians(twist); | ||
+ | this.rings = rings; | ||
+ | this.faces = faces; | ||
+ | points = new Vector3[faces * (rings+1)]; | ||
+ | update(); | ||
+ | } | ||
+ | | ||
+ | Vector3 getOutPos() { | ||
+ | return outPos.cpy(); | ||
+ | } | ||
+ | | ||
+ | Quaternion getOutOrientation() { | ||
+ | return outOrientation.cpy(); | ||
+ | } | ||
+ | |||
+ | public void update() { | ||
+ | int pointIdx = 0; | ||
+ | Quaternion segRot = new Quaternion(); | ||
+ | Quaternion rot = new Quaternion(); | ||
+ | Vector3 bendOffset = new Vector3(0.0, radInt + radiusIn, 0.0); | ||
+ | orientation.transform(bendOffset); | ||
+ | float angleSeg = 0.0; | ||
+ | for (int s=0; s<rings+1; s++) { | ||
+ | segRot.setFromAxisRad(zAxis, angleSeg); | ||
+ | segRot.mulLeft(orientation); | ||
+ | float tubeRadius = map(s, 0, rings, radiusIn, radiusOut); | ||
+ | Vector3 segOffset = new Vector3(0.0, radInt + tubeRadius, 0.0); | ||
+ | segRot.transform(segOffset); | ||
+ | angleSeg += bendAngle / rings; | ||
+ | float angle = s * twist/rings; | ||
+ | for (int i=0; i<faces; i++) { | ||
+ | tmpVec.set(0.0, tubeRadius, 0.0); | ||
+ | rot.setFromAxisRad(xAxis, angle); | ||
+ | rot.mulLeft(segRot); | ||
+ | rot.transform(tmpVec); | ||
+ | points[pointIdx++] = tmpVec.cpy().sub(segOffset).add(bendOffset).add(pos); | ||
+ | angle += TWO_PI / faces; | ||
+ | } | ||
+ | } | ||
+ | | ||
+ | outPos.set(0, -radiusOut - radInt, 0); | ||
+ | outOrientation.set(orientation); | ||
+ | outOrientation.mul(new Quaternion().setFromAxisRad(zAxis, bendAngle)); | ||
+ | outOrientation.transform(outPos); | ||
+ | outOrientation.mul(new Quaternion().setFromAxisRad(xAxis, twist)); | ||
+ | tmpVec.set(0.0, radInt + radiusIn, 0.0); | ||
+ | orientation.transform(tmpVec); | ||
+ | outPos.add(tmpVec).add(pos); | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | for (Vector3 p : points) { | ||
+ | point(p.x, p.y, p.z); | ||
+ | } | ||
+ | | ||
+ | for (int s=0; s<rings; s++) { | ||
+ | beginShape(QUAD_STRIP); | ||
+ | for (int i=0; i<faces+1; i++) { | ||
+ | tmpVec.set(points[s * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | tmpVec.set(points[(s+1) * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | } | ||
+ | endShape(); | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | </code> | ||
+ | </panel></accordion> | ||
+ | |||
+ | <accordion><panel title="scene.pde"> | ||
+ | <code java> | ||
+ | import queasycam.*; | ||
+ | import com.badlogic.gdx.math.*; | ||
+ | |||
+ | |||
+ | static final Vector3 xAxis = new Vector3(1.0, 0.0, 0.0); | ||
+ | static final Vector3 yAxis = new Vector3(0.0, 1.0, 0.0); | ||
+ | static final Vector3 zAxis = new Vector3(0.0, 0.0, 1.0); | ||
+ | |||
+ | |||
+ | QueasyCam cam; | ||
+ | |||
+ | boolean wireframe = false; | ||
+ | |||
+ | ArrayList<Drawable> objects = new ArrayList(); | ||
+ | |||
+ | |||
+ | public void setup() { | ||
+ | fullScreen(P3D); | ||
+ | |||
+ | cam = new QueasyCam(this, PConstants.PI/2.5f, 0.01, 9999); | ||
+ | cam.key_forward = 'z'; | ||
+ | cam.key_left = 'q'; | ||
+ | cam.key_backward = 's'; | ||
+ | cam.key_right = 'd'; | ||
+ | cam.key_up = 'e'; | ||
+ | cam.key_down = 'a'; | ||
+ | |||
+ | // Center tower | ||
+ | Pipe pipe = new Pipe( | ||
+ | new Vector3(0.0, 800, 0.0), | ||
+ | new Quaternion().setFromAxis(zAxis, -90), | ||
+ | 550, 64); | ||
+ | pipe.addTube(1000); | ||
+ | pipe.changeSize(800); | ||
+ | pipe.addTube(1000); | ||
+ | pipe.changeSize(820); | ||
+ | pipe.addTube(20); | ||
+ | pipe.changeSize(2600); | ||
+ | pipe.addTube(200); | ||
+ | pipe.addTube(-400); | ||
+ | objects.add(pipe); | ||
+ | |||
+ | // Surrounding circus | ||
+ | pipe = new Pipe( | ||
+ | new Vector3(0.0, 800, 0.0), | ||
+ | new Quaternion().setFromAxis(zAxis, -90), | ||
+ | 800, 64); | ||
+ | pipe.addTube(600); | ||
+ | pipe.changeSize(2000); | ||
+ | pipe.addTube(200); | ||
+ | pipe.changeSize(2040); | ||
+ | pipe.addTube(40); | ||
+ | pipe.addTube(1000); | ||
+ | objects.add(pipe); | ||
+ | |||
+ | for (int r=0; r<4; r++) { | ||
+ | Quaternion rot = new Quaternion().setEulerAngles(r * 90, 0, 9); | ||
+ | for (int i=0; i<6; i++) { | ||
+ | Vector3 pos = new Vector3(-2100, 298, (i-3)*34 + 17); | ||
+ | rot.transform(pos); | ||
+ | pipe = new Pipe(pos, rot, 16, 16); | ||
+ | pipe.addTube(1480 + 4 * abs(i-3)); | ||
+ | pipe.addBend(99, 180); | ||
+ | pipe.addTube(300, (3-i) * 5); | ||
+ | pipe.addBend(14); | ||
+ | if (abs(i-2.5)<=1) { | ||
+ | pipe.addTube(860); | ||
+ | pipe.changeSize(24); | ||
+ | pipe.addTube(18); | ||
+ | pipe.changeSize(26); | ||
+ | pipe.addTube(200); | ||
+ | } else { | ||
+ | pipe.addTube(1000); | ||
+ | pipe.addBend(70); | ||
+ | float l=0.0, ll=0.0; | ||
+ | int state = 0; // -1: left; 0: center; 1: right | ||
+ | while (l<1800) { | ||
+ | int turn = floor(random(3.0)-1.0); | ||
+ | if (state==0 && turn == -1) { | ||
+ | pipe.addBend(45, 90); | ||
+ | state = -1; | ||
+ | } else if (state==0 && turn == 1) { | ||
+ | pipe.addBend(45, -90); | ||
+ | state = 1; | ||
+ | } else if (state == -1 && turn == 1) { | ||
+ | pipe.addBend(45, 180, -90); | ||
+ | state = 0; | ||
+ | } else if (state == 1 && turn == -1) { | ||
+ | pipe.addBend(45, -180, 90); | ||
+ | state = 0; | ||
+ | } else if (state == 0) { | ||
+ | pipe.addTube(ll); | ||
+ | l += ll; | ||
+ | ll = 0.0; | ||
+ | } | ||
+ | ll += 100; | ||
+ | } | ||
+ | } | ||
+ | objects.add(pipe); | ||
+ | } | ||
+ | float deg = r * 90 + 45; | ||
+ | rot.setFromAxis(zAxis, -90); | ||
+ | rot.mul(new Quaternion().setFromAxis(xAxis, deg)); | ||
+ | Vector3 pos = new Vector3(590 * cos(radians(deg)), 800, 590 * sin(radians(deg))); | ||
+ | pipe = new Pipe(pos, rot, 34, 16); | ||
+ | pipe.addTube(980); | ||
+ | pipe.addBend(14); | ||
+ | pipe.addTube(950); | ||
+ | pipe.addBend(70); | ||
+ | pipe.addTube(1800); | ||
+ | objects.add(pipe); | ||
+ | | ||
+ | deg = r * 90 + 45; | ||
+ | rot.setFromAxis(zAxis, -90); | ||
+ | rot.mul(new Quaternion().setFromAxis(xAxis, deg)); | ||
+ | pos = new Vector3(1700 * cos(radians(deg)), 50, 1700 * sin(radians(deg))); | ||
+ | pipe = new Pipe(pos, rot, 140, 32); | ||
+ | pipe.addBend(90, 0, 0, 300); | ||
+ | objects.add(pipe); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | public void draw() { | ||
+ | background(0); | ||
+ | //lights(); | ||
+ | //ambientLight(12, 4, 1); | ||
+ | //pointLight(100, 100, 110, 140, -2000, 144); | ||
+ | directionalLight(204, 204, 204, .5, 0.6, 0.2); | ||
+ | emissive(50, 46, 51); | ||
+ | //specular(204, 102, 0); | ||
+ | shininess(1.0); | ||
+ | |||
+ | strokeWeight(1); | ||
+ | stroke(255, 0, 0); | ||
+ | line(0, 0, 0, 100, 0, 0); | ||
+ | stroke(0, 255, 0); | ||
+ | line(0, 0, 0, 0, 100, 0); | ||
+ | stroke(0, 0, 255); | ||
+ | line(0, 0, 0, 0, 0, 100); | ||
+ | |||
+ | if (wireframe) | ||
+ | stroke(0); | ||
+ | else | ||
+ | noStroke(); | ||
+ | strokeWeight(1); | ||
+ | |||
+ | for (Drawable o : objects) | ||
+ | o.draw(); | ||
+ | } | ||
+ | |||
+ | |||
+ | void keyPressed() { | ||
+ | if (key == 'w') | ||
+ | wireframe = !wireframe; | ||
+ | } | ||
+ | |||
+ | |||
+ | public interface Drawable { | ||
+ | public void draw(); | ||
+ | public void update(); | ||
+ | } | ||
+ | |||
+ | public interface Chainable extends Drawable { | ||
+ | public Vector3 getOutPos(); | ||
+ | public Quaternion getOutOrientation(); | ||
+ | } | ||
+ | |||
+ | |||
+ | /***************************** | ||
+ | ** class Pipe ** | ||
+ | *****************************/ | ||
+ | |||
+ | public class Pipe implements Drawable { | ||
+ | private Vector3 pos; | ||
+ | private Quaternion orientation; | ||
+ | private float radius; | ||
+ | private float nextRadius; | ||
+ | private int faces; | ||
+ | private ArrayList<Chainable> objects = new ArrayList(); | ||
+ | private Vector3 currentPos; | ||
+ | private Quaternion currentOrientation; | ||
+ | |||
+ | public Pipe(Vector3 pos, Quaternion orientation, float radius, int faces) { | ||
+ | this.pos = pos; | ||
+ | this.orientation = orientation; | ||
+ | this.radius = radius; | ||
+ | this.nextRadius = radius; | ||
+ | this.faces = faces; | ||
+ | currentPos = this.pos.cpy(); | ||
+ | currentOrientation = orientation.cpy(); | ||
+ | } | ||
+ | |||
+ | public void changeSize(float next) { | ||
+ | this.nextRadius = next; | ||
+ | } | ||
+ | | ||
+ | private void add(Chainable object) { | ||
+ | objects.add(object); | ||
+ | currentPos = object.getOutPos(); | ||
+ | currentOrientation = object.getOutOrientation(); | ||
+ | radius = nextRadius; | ||
+ | } | ||
+ | |||
+ | public void addTube(float len) { | ||
+ | addTube(len, 0.0); | ||
+ | } | ||
+ | | ||
+ | public void addTube(float len, float twist) { | ||
+ | Tube tube = new Tube( | ||
+ | currentPos, currentOrientation, | ||
+ | len, radius, nextRadius, twist, 1, faces | ||
+ | ); | ||
+ | add(tube); | ||
+ | } | ||
+ | | ||
+ | public void addBend(float angleBendDeg) { | ||
+ | addBend(angleBendDeg, 0.0, 0.0); | ||
+ | } | ||
+ | | ||
+ | public void addBend(float angleBendDeg, float angleRotDeg) { | ||
+ | addBend(angleBendDeg, angleRotDeg, 0.0); | ||
+ | } | ||
+ | | ||
+ | public void addBend(float angleBendDeg, float angleRotDeg, float twist) { | ||
+ | float radInt = 50; | ||
+ | addBend(angleBendDeg, angleRotDeg, twist, radInt); | ||
+ | } | ||
+ | |||
+ | public void addBend(float angleBendDeg, float angleRotDeg, float twist, float radInt) { | ||
+ | int rings = ceil(angleBendDeg * 64.0 / 360.0); | ||
+ | currentOrientation.mul(new Quaternion().setEulerAngles(0, angleRotDeg, 0)); | ||
+ | Bend bend = new Bend( | ||
+ | currentPos, currentOrientation, | ||
+ | radius, nextRadius, | ||
+ | radians(angleBendDeg), radInt, twist, | ||
+ | rings, faces | ||
+ | ); | ||
+ | add(bend); | ||
+ | } | ||
+ | |||
+ | public void update() { | ||
+ | for (Chainable o : objects) | ||
+ | o.update(); | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | for (Chainable o : objects) { | ||
+ | o.draw(); | ||
+ | } | ||
+ | |||
+ | Vector3 head = currentOrientation.transform(new Vector3(32, 0, 0)); | ||
+ | head.add(currentPos); | ||
+ | line(currentPos.x, currentPos.y, currentPos.z, head.x, head.y, head.z); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | /***************************** | ||
+ | ** class Tube ** | ||
+ | *****************************/ | ||
+ | |||
+ | public class Tube implements Drawable, Chainable { | ||
+ | private Vector3 pos; | ||
+ | private Quaternion orientation; | ||
+ | private float len; | ||
+ | private float radiusIn; | ||
+ | private float radiusOut; | ||
+ | private float twist; | ||
+ | private int rings; | ||
+ | private int faces; | ||
+ | private Vector3[] points; | ||
+ | private Vector3 tmpVec = new Vector3(); | ||
+ | private Vector3 outPos = new Vector3(); | ||
+ | private Quaternion outOrientation = new Quaternion(); | ||
+ | |||
+ | public Tube( | ||
+ | Vector3 pos, | ||
+ | Quaternion orientation, | ||
+ | float len, | ||
+ | float radiusIn, | ||
+ | float radiusOut, | ||
+ | float twist, // In degrees | ||
+ | int rings, int faces | ||
+ | ) { | ||
+ | this.pos = pos.cpy(); | ||
+ | this.orientation = orientation.cpy(); | ||
+ | this.len = len; | ||
+ | this.radiusIn = radiusIn; | ||
+ | this.radiusOut = radiusOut; | ||
+ | this.twist = radians(twist); | ||
+ | this.rings = rings; | ||
+ | this.faces = faces; | ||
+ | points = new Vector3[faces * (rings+1)]; | ||
+ | update(); | ||
+ | } | ||
+ | | ||
+ | Vector3 getOutPos() { | ||
+ | return outPos.cpy(); | ||
+ | } | ||
+ | | ||
+ | Quaternion getOutOrientation() { | ||
+ | return outOrientation.cpy(); | ||
+ | } | ||
+ | | ||
+ | public void update() { | ||
+ | Quaternion rot = new Quaternion(); | ||
+ | int pointIdx = 0; | ||
+ | for (int s=0; s<rings+1; s++) { | ||
+ | float angle = s * twist/rings; | ||
+ | for (int i=0; i<faces; i++) { | ||
+ | tmpVec.set(s * len/rings, map(s, 0, rings, radiusIn, radiusOut), 0.0); | ||
+ | rot.setFromAxisRad(xAxis, angle).mulLeft(orientation); | ||
+ | rot.transform(tmpVec); | ||
+ | points[pointIdx++] = tmpVec.cpy().add(pos); | ||
+ | angle += TWO_PI / faces; | ||
+ | } | ||
+ | } | ||
+ | outOrientation.set(orientation); | ||
+ | outOrientation.mul(new Quaternion().setFromAxisRad(xAxis, twist)); | ||
+ | outPos.set(len, 0.0, 0.0); | ||
+ | outOrientation.transform(outPos); | ||
+ | outPos.add(pos); | ||
+ | | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | for (int s=0; s<rings; s++) { | ||
+ | beginShape(QUAD_STRIP); | ||
+ | for (int i=0; i<faces+1; i++) { | ||
+ | tmpVec.set(points[s * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | tmpVec.set(points[(s+1) * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | } | ||
+ | endShape(); | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | |||
+ | |||
+ | /***************************** | ||
+ | ** class Bend ** | ||
+ | *****************************/ | ||
+ | |||
+ | public class Bend implements Drawable, Chainable { | ||
+ | private Vector3 pos; | ||
+ | private Quaternion orientation; | ||
+ | private float radiusIn; | ||
+ | private float radiusOut; | ||
+ | private float bendAngle; | ||
+ | private float radInt; | ||
+ | private float twist; | ||
+ | private int rings; | ||
+ | private int faces; | ||
+ | private Vector3[] points; | ||
+ | private Vector3 tmpVec = new Vector3(); | ||
+ | private Vector3 outPos = new Vector3(); | ||
+ | private Quaternion outOrientation = new Quaternion(); | ||
+ | |||
+ | public Bend( | ||
+ | Vector3 pos, | ||
+ | Quaternion orientation, | ||
+ | float radiusIn, | ||
+ | float radiusOut, | ||
+ | float bendAngle, // In radians | ||
+ | float radInt, // Internal radius | ||
+ | float twist, | ||
+ | int rings, | ||
+ | int faces | ||
+ | ) { | ||
+ | this.pos = pos.cpy(); | ||
+ | this.orientation = orientation.cpy(); | ||
+ | this.radiusIn = radiusIn; | ||
+ | this.radiusOut = radiusOut; | ||
+ | this.bendAngle = bendAngle; | ||
+ | this.radInt = radInt; | ||
+ | this.twist = radians(twist); | ||
+ | this.rings = rings; | ||
+ | this.faces = faces; | ||
+ | points = new Vector3[faces * (rings+1)]; | ||
+ | update(); | ||
+ | } | ||
+ | | ||
+ | Vector3 getOutPos() { | ||
+ | return outPos.cpy(); | ||
+ | } | ||
+ | | ||
+ | Quaternion getOutOrientation() { | ||
+ | return outOrientation.cpy(); | ||
+ | } | ||
+ | |||
+ | public void update() { | ||
+ | int pointIdx = 0; | ||
+ | Quaternion segRot = new Quaternion(); | ||
+ | Quaternion rot = new Quaternion(); | ||
+ | Vector3 bendOffset = new Vector3(0.0, radInt + radiusIn, 0.0); | ||
+ | orientation.transform(bendOffset); | ||
+ | float angleSeg = 0.0; | ||
+ | for (int s=0; s<rings+1; s++) { | ||
+ | segRot.setFromAxisRad(zAxis, angleSeg); | ||
+ | segRot.mulLeft(orientation); | ||
+ | float tubeRadius = map(s, 0, rings, radiusIn, radiusOut); | ||
+ | Vector3 segOffset = new Vector3(0.0, radInt + tubeRadius, 0.0); | ||
+ | segRot.transform(segOffset); | ||
+ | angleSeg += bendAngle / rings; | ||
+ | float angle = s * twist/rings; | ||
+ | for (int i=0; i<faces; i++) { | ||
+ | tmpVec.set(0.0, tubeRadius, 0.0); | ||
+ | rot.setFromAxisRad(xAxis, angle); | ||
+ | rot.mulLeft(segRot); | ||
+ | rot.transform(tmpVec); | ||
+ | points[pointIdx++] = tmpVec.cpy().sub(segOffset).add(bendOffset).add(pos); | ||
+ | angle += TWO_PI / faces; | ||
+ | } | ||
+ | } | ||
+ | | ||
+ | outPos.set(0, -radiusOut - radInt, 0); | ||
+ | outOrientation.set(orientation); | ||
+ | outOrientation.mul(new Quaternion().setFromAxisRad(zAxis, bendAngle)); | ||
+ | outOrientation.transform(outPos); | ||
+ | outOrientation.mul(new Quaternion().setFromAxisRad(xAxis, twist)); | ||
+ | tmpVec.set(0.0, radInt + radiusIn, 0.0); | ||
+ | orientation.transform(tmpVec); | ||
+ | outPos.add(tmpVec).add(pos); | ||
+ | } | ||
+ | |||
+ | public void draw() { | ||
+ | for (Vector3 p : points) { | ||
+ | point(p.x, p.y, p.z); | ||
+ | } | ||
+ | | ||
+ | for (int s=0; s<rings; s++) { | ||
+ | beginShape(QUAD_STRIP); | ||
+ | for (int i=0; i<faces+1; i++) { | ||
+ | tmpVec.set(points[s * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | tmpVec.set(points[(s+1) * faces + i%faces]); | ||
+ | vertex(tmpVec.x, tmpVec.y, tmpVec.z); | ||
+ | } | ||
+ | endShape(); | ||
+ | } | ||
} | } | ||
} | } | ||
</code> | </code> | ||
</panel></accordion> | </panel></accordion> |