Bruit de Perlin

Formation

En Ligne

Prix sur demande

Appeler le centre

Avez-vous besoin d'un coach de formation?

Il vous aidera à comparer différents cours et à trouver la solution la plus abordable.

Description

  • Typologie

    Formation

  • Méthodologie

    En ligne

Grâce à cette formation vous pourrez acquérir les connaissances nécessaires qui vous permettrons d’ajouter des compétences à votre profil et obtenir de solides aptitude qui vous offriront de nombreuses opportunités professionnelles.

Questions / Réponses

Ajoutez votre question

Nos conseillers et autres utilisateurs pourront vous répondre

À qui souhaitez-vous addresser votre question?

Saisissez vos coordonnées pour recevoir une réponse

Nous ne publierons que votre nom et votre question

Les Avis

Le programme

Introduction du cours

Ce tutoriel a pour but de vous initier à la notion de "fonction de bruit" en informatique, et ensuite de vous faire comprendre le fonctionnement d'une d'entre elles, le bruit de Perlin. Ces fonctions mathématiques sont très utiles notamment dans l'art numérique, et notamment les effets spéciaux dans le cinéma, les jeux vidéos, et j'en passe.

La première fonction de ce genre a été inventé par Ken Perlin pour donner plus de réalisme aux premières images de synthèse. A l'époque, la souris n'en n'était qu'à ses balbutiements, la création des objets (notamment pour le film Tron de 1982) se faisait dans des éditeurs de textes, et les objets étaient des assemblages de formes de base (cube, sphère, etc.). Les éditeurs d'images n'existaient pas encore, et les textures non plus ! Le bruit de Perlin a eu pour vocation de créer automatiquement des textures, pour rajouter du désordre et leur donner un peu plus de réalisme. Les applications à Perlin se sont depuis très largement diversifiées, le bruit de Perlin peut être utilisé dans de très nombreuses situations, la seule limite est votre imagination.

Ce bruit constitue la base pour la génération procédurale d'objets complexes, tels que des arbres, des skybox ou même des planètes entières !

Fonction de bruit ?

Une fonction de bruit... Quésako ?

Séparons cette expression en deux.
Que fait une simple fonction mathématique ? Elle possède un certain nombre de paramètres d'entrée, fait des calculs à l'aide de ces paramètres et renvoie une (ou plusieurs) valeur(s) en sortie. Dans un contexte de programmation, c'est exactement la même chose, excepté qu'une fonction ne renverra qu'une seule valeur en sortie.

Maintenant, qu'est ce qu'un bruit ? Dans la vie courante, ce terme représente un grand nombre d'effets, tels "la neige" sur votre téléviseur, le grésillement des enceintes de votre chaîne Hi-Fi, le brouhaha dans les endroits bondés de monde. En informatique cependant, le bruit n'as pas une connotation négative, bien au contraire. C'est le terme utilisé lorsque l'on cherche à simuler de l'aléatoire.

Pourquoi simuler l'aléatoire ?

Tout simplement car notre monde réel l'est, et par conséquent un jeu vidéo ou une image de synthèse seront d'autant plus crédibles si ils contiennent des éléments aux propriétés aléatoires.

Si vous êtes bon observateur, vous avez peut être remarqué une autre caractéristique dans le monde réel : la cohérence. Les nuages dans le ciel par exemple, ont tous des formes différentes, mais présente des caractéristiques communes, leur couleur par exemple. Ils ne sont pas blanc et violet et orange, non, ils sont blancs avec de légères variations. Ils ont donc des caractéristiques aléatoires mais aussi cohérentes.

Les fonctions de bruit sont donc des fonctions mathématiques qui permettront de simuler de l'aléatoire, avec la possibilité d'avoir une cohérence dans ce bruit.

Sachez qu'il est très facile de générer de l'aléatoire incohérent (cela revient à générer une suite de nombre), mais qu'il est un peu plus difficile de générer de l'aléatoire cohérent. C'est à ça que sert l'algorithme de Perlin que je vais présenter dans le chapitre suivant.

Exemple d'application

L'application la plus populaire d'une fonction de bruit est la génération d'un terrain. En informatique, un terrain est simplement un assemblage de triangles dont on modifie l'altitude de chaque point, en fonction d'une image de hauteur (appelée communément heightmap, qui signifie carte d'altitude).

Heightmap

En superposant le maillage du terrain (pour l'instant totalement plat) et la heightmap, on peut trouver à quel pixel correspond chaque sommet des triangles du terrain. Il est alors aisé de donner à ce sommet une hauteur proportionnelle à la valeur de gris du pixel.
En appliquant cette méthode sur tout les points, vous obtenez un terrain correspondant à cette heightmap.

Pour vous faire une idée du maillage du terrain avant/après :

avant déformation

après déformation

Ce terrain a belle allure, mais j'ai omis un détail. Il n'a pas été généré avec une heightmap mais avec une fonction de bruit ! La fonction de bruit permet donc d'obtenir le même résultat, mais présente quelques avantages supplémentaires.
Pour obtenir ce terrain, j'ai utilisé une fonction de bruit appelée NOISE dont voici la forme :

réel NOISE(réel x, réel y)

Les coordonnées (x,y) envoyées en paramètres de la fonction correspondent aux coordonnées d'un point P quelconque. La fonction renvoie en sortie un nombre réel, qui correspond ici à l'altitude à donner au point P. Il faut donc déformer tous les points du terrain un par un en appelant cette fonction pour chacun d'entre eux pour obtenir le terrain final.

Vous verrez comment fonctionne NOISE dans la prochaine partie. Cette fonction possède plusieurs propriétés intéressantes :

  • Elle génère un bruit simple qui simule l'aléatoire, et pour un point (x,y) donné, elle renverra toujours la même valeur.

  • Elle génère un bruit cohérent.

  • Utilisation de ressources : elle utilise beaucoup moins de mémoire qu'une heightmap, et présente un niveau de précision identique (si vous mettez en place une méthode d'interpolation sur l'image).

  • Espace infini : la fonction peut être utilisée pour n'importe quelle valeur de (x,y). Un détail cependant, le bruit se répète au bout de 256 unités, mais dans la pratique ce défaut est invisible (et il peut même être tourné en avantage si l'on veut avoir une image raccordable).

Principe/Algorithme

L'algorithme que je vais vous présenter fonctionne en deux dimensions mais est très facilement extensible en 3D,4D et même bien plus.

Tout d'abord, imaginez une grille infinie qui subdivise l'espace en cases carrées de largeur unitaire. Les intersections des lignes et des colonnes seront des points donc les coordonnées peuvent être exprimées avec des nombres entiers.

A chaque fois que la fonction sera appelée pour un couple (x,y), elle déterminera la position des 4 sommets des carrés les plus proches.

Si on prends un point P au hasard, par exemple P(5.5,4.5) ça nous donne le schéma suivant :

Visuellement on voit bien que le point P est entouré des sommets (5.0,4.0) (6.0,4.0) (6.0,5.0) (5.0,5.0). Il suffit donc d'enlever la partie fractionnaire des coordonnées de P pour obtenir facilement les coordonnées des quatre sommets.

Maintenant, à chacun des 4 sommets, la fonction va associer pseudo-aléatoirement un vecteur 2D de longueur unitaire.

Tableau de gradients

Tout d'abord on créé un tableau contenant un ensemble de directions. Le tableau contient au total 8 vecteurs 2D de longueur unitaire :

unit = 1.0/sqrt(2) gradient2[8][2] = {{unit,unit},{-unit,unit},{unit,-unit},{-unit,-unit},                    {1,0},      {-1,0},      {0,1},       {0,-1}}

Table de permutation

Le second tableau est véritablement la colonne vertébrale de votre fonction de bruit. C'est une table de permutation contenant une seule fois chaque nombre de 0 à 255, mis dans le désordre. Pour changer le bruit généré, il suffira de mélanger à nouveau cette table.

Table de permutation ? Quesako ?

Ce terme peut avoir de nombreuses significations. Ici, c'est avant tout un tableau de nombres que nous allons utiliser d'une façon un peu particulière pour simuler du désordre (du pseudo-aléatoire).

Le seul critère ici est que la table soit assez bien mélangée. En voici une toute prête :

perm[256] =  {  151,160,137,91,90,15,131,13,201,95,96,53,194,233,7,225,140,36,103,30,69,  142,8,99,37,240,21,10,23,190, 6,148,247,120,234,75,0,26,197,62,94,252,219,  203,117,35,11,32,57,177,33,88,237,149,56,87,174,20,125,136,171,168, 68,175,  74,165,71,134,139,48,27,166,77,146,158,231,83,111,229,122,60,211,133,230,220,  105,92,41,55,46,245,40,244,102,143,54, 65,25,63,161,1,216,80,73,209,76,132,  187,208, 89,18,169,200,196,135,130,116,188,159,86,164,100,109,198,173,186,3,  64,52,217,226,250,124,123,5,202,38,147,118,126,255,82,85,212,207,206,59,227,  47,16,58,17,182,189,28,42,223,183,170,213,119,248,152, 2,44,154,163, 70,221,  153,101,155,167, 43,172,9,129,22,39,253, 19,98,108,110,79,113,224,232,178,185,  112,104,218,246,97,228,251,34,242,193,238,210,144,12,191,179,162,241,81,51,145,  235,249,14,239,107,49,192,214, 31,181,199,106,157,184, 84,204,176,115,121,50,45,  127, 4,150,254,138,236,205,93,222,114,67,29,24,72,243,141,128,195,78,66,215,61,  156,180 }

Pour simplifier le code par la suite, je n'utilise pas directement cette table mais un tableau de 512 cases contenant deux fois celle-ci, une première fois de l'indice 0 à 255, puis une deuxième fois de l'indice 256 à 511. Le petit algorithme que j'utilise pour faire la copie :

for i = 0 to 511     permtable[i] = perm[i & 255]

Le "&" est un ET logique, c'est le symbole utilisé pour réaliser cette opération en C et C++ et probablement aussi dans votre language favori. (i & 255) donne toujours une valeur comprise dans [0;255]

Rappelez vous, nous voulons associer un des vecteurs du premier tableau à chacun des quatre sommets du carré englobant. Voici comment on procède pour la première dont les coordonnées sont (x0,y0) et la dernière (x0+1,y0+1) :

//On fait un masquage, ii et jj sont compris entre 0 et 255 ii = x0 & 255 jj = y0 & 255 //Une manière un peu particulière de créer du désordre // Le modulo (%) 8 limite les valeurs de grad1 et grad4 entre 0 et 7 grad1 = permtable[ii + permtable[jj]] % 8 grad4 = permtable[ii + 1 + permtable[jj + 1]] % 8 //On récupère simplement les valeurs des vecteurs vecteurPremier.x = gradient2[grad1][0] vecteurPremier.y = gradient2[grad1][1] vecteurDernier.x = gradient2[grad4][0] vecteurDernier.y = gradient2[grad4][1]

Le % est l'opérateur modulo, a % b donne tout simplement le reste de la division euclidienne de a par b. Ce nombre est donc toujours inférieur à b

Alors que fait ce pseudo-code ? La seule difficulté réside dans les lignes 6 et 7. Rappelez vous, la table contient des nombres en désordre. On utilise une première fois la table pour obtenir un nombre au hasard. On additionne ce nombre avec ii, et on utilise cette nouvelle valeur une fois de plus dans la table pour obtenir la valeur finale du sommet. Cette valeur est ensuite utilisée comme indice dans le premier tableau de 8 vecteurs.

Si vous avez bien compris l'idée, bravo, sinon pas la peine de s'inquiéter, ça viendra tout seul avec le temps.

Il nous manque encore les valeurs associées aux deux autres sommets, prenez deux minutes pour y réfléchir et la formule devrait venir toute seule :

grad2 = permtable[ii + permtable[jj + 1]] % 8 grad3 = permtable[ii + 1 + permtable[jj]] % 8 vecteurDeuxieme.x = gradient2[grad2][0] vecteurDeuxieme.y = gradient2[grad2][1] vecteurTroisieme.x = gradient2[grad3][0] vecteurTroisieme.y = gradient2[grad3][1]

Note : Cette façon de faire garantit que le bruit renverra toujours les mêmes valeurs pour un couple (x,y) identique, mais aussi que le bruit final paraîtra assez aléatoire.

Maintenant que nous avons ces quatre valeurs, il nous faut trouver la valeur associée à P. Un petit schéma pour résumer :

Ici, l'idée est simple. Chaque sommet va contribuer à la valeur finale de P.

Plus un vecteur d'un sommet va pointer vers P, plus sa contribution sera grande, donc plus la hauteur de P va augmenter (sur une image, plus la couleur du pixel P sera proche du blanc). A l'inverse, si un vecteur pointe plutôt à l'opposé, sa contribution sera faible.

Second critère, plus P sera proche d'un sommet, plus il sera sensible à la contribution de ce sommet. Cette étape est appelée interpolation.

Contributions

On commence par calculer les 4 contributions :

//(x,y) représente les coordonnées du point P s = gradA . ((x, y) - (x0, y0)) //(x0,y0) = coordonnées du point A t = gradD . ((x, y) - (x0, y0+1)) //(x0,y0+1) = coordonnées du point D u = gradB . ((x, y) - (x0+1, y0)) //etc. v = gradC . ((x, y) - (x0+1, y0+1))

Le "." est le produit scalaire, c'est un simple opérateur mathématique sur des vecteurs. Le produit scalaire de deux vecteurs V1(a,b) et V2(c,d) est donné par la formule :

V1.V2 = ac + bd

Interpolation

Maintenant on va pondérer les contributions, en réalisant une interpolation. Pour obtenir la hauteur finale, on va interpoler ces valeurs deux par deux selon l'axe des abscisses (x), puis les deux valeurs intermédiaires selon l'axe des ordonnées.

//on calcule le coefficient d'interpolation (selon x) Sx = 3*(x-x0)*(x-x0) - 2*(x-x0)*(x-x0)*(x-x0) //on lisse les valeurs 2 à 2 stLisse = s + Sx*(t-s) uvLisse = u + Sx*(v-u) //on calcule le 2ème coefficient d'interpolation (selon y) Sy = 3*(y-y0)*(y-y0) - 2*(y-y0)*(y-y0)*(y-y0) //Le résultat final valeurFinale =  stLisse + Sy*(uvLisse-stLisse)

C'est une méthode simplissime d'interpolation, cela ne devrait pas poser de problème. Pour lisser les valeurs, j'ai utilisé la fonction $y = 3x^{2}-2x^{3}{$. Les résultats sont beaucoup plus doux qu'avec une interpolation linéaire, et la fonction vaut quand même zéro pour x=0 et 1 pour x=1.

(Image tracée avec ZeGrapher)

Pour résumer tout ça, rien de tel qu'un code complet, codé en C :

float Get2DPerlinNoiseValue(float x, float y, float res) { float tempX,tempY; int x0,y0,ii,jj,gi0,gi1,gi2,gi3; float unit = 1.0f/sqrt(2); float tmp,s,t,u,v,Cx,Cy,Li1,Li2; float gradient2[][2] = {{unit,unit},{-unit,unit},{unit,-unit},{-unit,-unit},{1,0},{-1,0},{0,1},{0,-1}}; unsigned int perm[] = {151,160,137,91,90,15,131,13,201,95,96,53,194,233,7,225,140,36,103,30,69, 142,8,99,37,240,21,10,23,190,6,148,247,120,234,75,0,26,197,62,94,252,219, 203,117,35,11,32,57,177,33,88,237,149,56,87,174,20,125,136,171,168,68,175, 74,165,71,134,139,48,27,166,77,146,158,231,83,111,229,122,60,211,133,230,220, 105,92,41,55,46,245,40,244,102,143,54,65,25,63,161,1,216,80,73,209,76,132, 187,208,89,18,169,200,196,135,130,116,188,159,86,164,100,109,198,173,186,3, 64,52,217,226,250,124,123,5,202,38,147,118,126,255,82,85,212,207,206,59,227, 47,16,58,17,182,189,28,42,223,183,170,213,119,248,152,2,44,154,163,70,221, 153,101,155,167,43,172,9,129,22,39,253,19,98,108,110,79,113,224,232,178,185, 112,104,218,246,97,228,251,34,242,193,238,210,144,12,191,179,162,241,81,51,145, 235,249,14,239,107,49,192,214,31,181,199,106,157,184,84,204,176,115,121,50,45, 127,4,150,254,138,236,205,93,222,114,67,29,24,72,243,141,128,195,78,66,215,61, 156,180}; //Adapter pour la résolution x /= res; y /= res; //On récupère les positions de la grille associée à (x,y) x0 = (int)(x); y0 = (int)(y); //Masquage ii = x0 & 255; jj = y0 & 255; //Pour récupérer les vecteurs gi0 = perm[ii + perm[jj]] % 8; gi1 = perm[ii + 1 + perm[jj]] % 8; gi2 = perm[ii + perm[jj + 1]] % 8; gi3 = perm[ii + 1 + perm[jj + 1]] % 8; //on récupère les vecteurs et on pondère tempX = x-x0; tempY = y-y0; s = gradient2[gi0][0]*tempX + gradient2[gi0][1]*tempY; tempX = x-(x0+1); ...

Appeler le centre

Avez-vous besoin d'un coach de formation?

Il vous aidera à comparer différents cours et à trouver la solution la plus abordable.

Bruit de Perlin

Prix sur demande