- Créer un sous-dossier «TP08» dans le dossier «TP_informatique_commune» de votre repertoire personnel
- Si on vous demande d'écrire un script python à la question 3\c) de l'activité 2 du TP 7 vous enregistrez ce script dans un fichier nommé «tp07_act02_q03c.py». Remarquez que l'on utilisera toujours deux chiffres décimaux pour représenter les entiers et que l'on va toujours du général vers le particulier car ainsi l'ordre alphabétique correspond à l'ordre tri chronologique.
- On rendra son code modulaire en utilisant systématiquement des fonctions
- On testera toutes ses fonctions
1. Manipulation d'images
Le module matplotlib
permet de manipuler des images sous la forme
de tableaux à deux dimensions (c'est à dire sous la forme de listes
de listes). Pour tout ce TP on fournit les fonctions ci-dessous (à
copier en haut de vos scripts)
import matplotlib.pyplot as plt import numpy as np def plot_as_image(rect_array): """Affiche l'image correspondant au tableau rect_array""" plt.figure() plt.axis('off') plt.figimage(np.array(rect_array)) plt.show() def plot_as_gray_image(rect_array): """Affiche l'image correspondant au tableau rect_array""" plt.figure() plt.axis('off') plt.figimage(np.array(rect_array), cmap=plt.gray()) plt.show() def image_to_array(chemin_image): """Retourne un tableau correspondant à l'image de chemin d'accès 'chemin_image' """ return list(plt.imread(chemin_image))
Dans tout ce TP nous travaillerons sur des images au format png du
tableau "Le Cri" (Skrik)) du peintre expressionniste norvégien
Edvard Munch. Ces dernieres sont à télécharger ci-dessous en
téléchargant les fichiers dans un sous-dossier nommé img
du
dossier de votre TP (ainsi tous les script de ce TP doivent être
dans le même dossier du sous-dossier "img" qui lui-même contient les
fichiers munch_le_cri.png
et munch_le_cri_NB.png
).
Dans la suite on supposera que cette image est sauvergardée dans le dossier de ce TP (avec le même nom munch_le_cri.png). On supposera de plus que la ligne suivante est toujours présente en haut de vos scripts
import os img_fpath = os.path.join("img", "munch_le_cri.png") img_NB_fpath = os.path.join("img", "munch_le_cri_NB.png")
En résumé on commencera ses scripts par
import matplotlib.pyplot as plt import numpy as np def plot_as_image(rect_array): """Affiche l'image correspondant au tableau rect_array""" plt.figure() plt.axis('off') plt.figimage(np.array(rect_array)) plt.show() def plot_as_gray_image(rect_array): """Affiche l'image correspondant au tableau rect_array""" plt.figure() plt.axis('off') plt.figimage(np.array(rect_array), cmap=plt.gray()) plt.show() def image_to_array(chemin_image): """Retourne un tableau correspondant à l'image de chemin d'accès 'chemin_image' """ return list(plt.imread(chemin_image)) import os img_fpath = os.path.join("img", "munch_le_cri.png") img_NB_fpath = os.path.join("img", "munch_le_cri_NB.png")
Vérifier que le programme suivant affiche une image sinon appeler le professeur (rappel: comme écrit ci-dessus vous devez enregistrer l'image nommée
munch_le_cri.png
dans un dossier nomméimg
du dossier contenant votre script)import matplotlib.pyplot as plt import numpy as np def plot_as_image(rect_array): """Affiche l'image correspondant au tableau rect_array""" plt.figure() plt.axis('off') plt.figimage(np.array(rect_array)) plt.show() def plot_as_gray_image(rect_array): """Affiche l'image correspondant au tableau rect_array""" plt.figure() plt.axis('off') plt.figimage(np.array(rect_array), cmap=plt.gray()) plt.show() def image_to_array(chemin_image): """Retourne un tableau correspondant à l'image de chemin d'accès 'chemin_image' """ return list(plt.imread(chemin_image)) import os img_fpath = os.path.join("img", "munch_le_cri.png") img_NB_fpath = os.path.join("img", "munch_le_cri_NB.png") # Rappel: les lignes précédentes devront toujours être ajoutées en # début de script dans ce TP image_as_array = image_to_array("./img/munch_le_cri.png") plot_as_image(image_as_array) plt.show()
Vérifier que le programme suivant affiche une image (sinon appeler le professeur)
L = [[0, 0, 0, 0, 0], [0, 1, 1, 1, 1], [0, 1, 2, 2, 2], [0, 1, 2, 3, 3], [0, 1, 2, 3, 4]] plot_as_gray_image(L)
Vérifier que le module
matplotlib
effectue "une balance des blancs" automatique en vérifiant que le script ci-dessous affiche la même image que précédemmentL = [[0, 0, 0, 0, 0], [0, 1, 1, 1, 1], [0, 1, 2, 2, 2], [0, 1, 2, 3, 3], [0, 1, 2, 3, 4]] # Déformation affine de l'intensité: # on multiplie ar 2 puis on ajoute 3 à l'intensité de chaque pixel L_deformation_affine = [[2*color+3 for color in line] for line in L] plot_as_gray_image(L_deformation_affine)
Une image couleur est représentée par un tableau de dimension trois (càd une liste de listes de listes représentant une liste de lignes chacune étant une liste de pixel chacun étant une liste de trois flottants compris entre \(0\) et \(1\) représentant chacun une composante primaire de la couleur du pixel) de flottants compris entre 0 et 1. À l'aide du script suivant déterminer l'ordre correct des trois canaux de couleur: est-ce RVB, RBV, BRV, BVR, RVB ou RBV?
L = [[[1., 0., 0.], [1., 0., 0.], [1., 0., 0.]], [[0., 1., 0.], [0., 1., 0.], [0., 1., 0.]], [[0., 0., 1.], [0., 0., 1.], [0., 0., 1.]], [[1., 1., 1.], [1., 1., 1.], [1., 1., 1.]], [[0., 1., 1.], [0., 1., 1.], [0., 1., 1.]], [[1., 0., 1.], [1., 0., 1.], [1., 0., 1.]], [[1., 1., 0.], [1., 1., 0.], [1., 1., 0.]]] plot_as_image(L)
- Soit une image couleur représentée par un tableau
L
. Dans l'expressionL[i][j][k]
quel indice parmii
,j
ouk
est- un indice ligne?
- un indice colonne?
- un indice de canal (=couleur)?
- Si
tab
est un tableau de dimension \(3\) de taille \(n\times{}p\times{}3\) représentant une image couleur, dire en français à quoi correspond l'expressions suivante (on pourra écrire une phrase sur le modèle de la suivante: «cette expression correspond à la composante rouge/verte/bleue du pixel situé en … de l'image»).tab[0][0][0]
?tab[-1][0][1]
?tab[0][-1][2]
?tab[-1][-1][-1]
?
- Soit une image couleur représentée par un tableau
Compléter les fonctions
symetrie_axe_vertical_v1
etsymetrie_axe_vertical_v2
qui prennent en argument une image (sous la forme d'un tableau) et retourne une nouvelle image obtenue par symétrie autour d’un axe vertical passant par le milieu de l’image (donc la gauche passe à droite et réciproquement).def symetrie_axe_vertical_v1(im): n, p = len(??), len(??) im_res = ?? for i in range(??): im_res.append(im[??]) return ??
def symetrie_axe_vertical_v2(im): n, p = len(??), len(??) return [im[??] for i in ??] # liste définie par compréhension
Ainsi chacun des scripts suivants doit afficher l'image qui lui suit (images identiques évidemment)
im = image_to_array(img_fpath) im_sym = symetrie_axe_vertical_v1(im) plot_as_image(im_sym)
im = image_to_array(img_fpath) im_sym = symetrie_axe_vertical_v2(im) plot_as_image(im_sym)
Écrire une fonction
symetrie_axe_horizontale
qui prend en argument une image (sous la forme d'un tableau de dimension) et retourne une nouvelle image, obtenue par symétrie autour d’un axe horizontale passant par le milieu de l’image. Ainsi les lignes suivantes doivent afficher l'image qui suitRédiger une fonction negatif qui prend en argument une image couleur et retourne son négatif, c’est-à-dire l’image dans laquelle chaque composante de couleur
c
de chaque pixel est remplacée par sa valeur complémentaire1-c
. Remarque: éviter d'utiliser des indices, ici vous n'en avez pas besoin.Écrire une fonction
split_color_channels(imtab)
qui prend en argument un tableau numpyimtab
représentant une image couleur et retourne un triplet de tableaux numpy de même taille représentant dans l'ordre la couleur rouge, la couleur verte et la couleur bleue. Pour afficher les trois images côte à côte on utilisera le code suivant# Créer une figure constituée de trois repères disposés en 3 colonnes # et une ligne fig, axes = plt.subplots(ncols=3, nrows=1) # Afficher chaque image dans son repère axes[0].imshow(image_rouge) # axes[0] est le repère de gauche axes[1].imshow(image_verte) # axes[1] est le repère du milieu axes[2].imshow(image_bleue) # axes[2] est le repère de droite
Compléter la fonctions
color_to_grayscale_simple
qui accepte un tableau représentant une image couleur en argument et qui retourne un tableau représentant une image en noir et blanc (donc avec une dimension en moins) en faisant simplement la moyenne des trois composantes de couleur.def color_to_grayscale_simple(im): n, p = len(??), len(??) im_res = ?? for row in ??: new_row = ?? for pixel_colors in ??: pix_intensity = ?? ??.append(??) return ??
Compléter la fonctions
color_to_grayscale
qui accepte un tableau représentant une image couleur en argument et qui retourne un tableau représentant une image en noir et blanc (donc avec une dimension en moins) en utilisant la formule \(I=0.299\times{}R + 0.587\times{}G + 0.114\times{}B\)def color_to_grayscale(im): n, p = len(??), len(??) im_res = ?? for row in ??: new_row = ?? for pixel_colors in ??: R, G, B = ?? pix_intensity = ?? ??.append(??) return ??
(Passez plutôt à la section suivante) Rédiger une fonction
appliquer_fonction(imtab, fun)
qui prend en argument un tableau numpyimtab
représentant une image et retourne un nouveau tableau obtenu en appliquant la fonctionfun
à tous les pixel de l'imageimtab
(chaque pixel étant donc une liste de trois flottants compris entre \(0\) et \(1\)). Vous devriez obtenir l'image ci-dessous.img/munch_le_cri_negative.png
2. Traitement d'images
Une première activité introductive
Écrire une fonction
floutage_unif_voisins_immediats
qui accepte un tableau de taille notée \(n\times{}p\) représentant une image noir et blanc (pour simplifier) en argument et qui retourne un tableau représentant une image en noir et blanc de taille \((n-2)\times{}(p-2)\) représantant les pixels intérieurs (non situées au bord) dont l'intensité a été moyennée avec celle de ses \(4\) voisins directs (donc diagonale non comprise).Écrire une fonction
convolution_sharpen_kernel
qui accepte un tableau de taille notée \(n\times{}p\) représentant une image noir et blanc (pour simplifier) en argument et qui retourne un tableau représentant une image en noir et blanc de taille \((n-2)\times{}(p-2)\) représantant les pixels intérieurs (non situées au bord) dont l'intensité \(I_{i,j}\) est remplacée par \(I'_{i,j}=5\times{}I_{i,j}-I_{i-1,j}-I_{i+1,j}-I_{i,j-1}-I_{i,j+1}\).Écrire une fonction
convolution_ysobel_kernel
qui accepte un tableau de taille notée \(n\times{}p\) représentant une image noir et blanc (pour simplifier) en argument et qui retourne un tableau représentant une image en noir et blanc de taille \((n-2)\times{}(p-2)\) représantant les pixels intérieurs (non situées au bord) dont l'intensité \(I_{i,j}\) est remplacée par \(I'_{i,j}=I_{i-1,j-1}+2*I_{i,j-1}+I_{i+1,j-1} -I_{i-1,j+1}-2*I_{i,j+1}-I_{i+1,j+1}\).
Nous souhaitons maintenant généraliser tous les traitements précédents et traiter nos images en leur appliquant des masques linéaires c'est à dire en remplaçant la valeur de chaque pixel par une combinaison linéaire de la valeur des pixels qui lui sont adjacents.
Dans un premier temps Tous nos masques seront de taille \(3x3\) et
seront donc représentés par une matrice de taille \(3x3\) comme
ci-contre \(C = \begin{pmatrix} c_{11}&c_{12}&c_{13}
\\ c_{21}&c_{22}&c_{23} \\ c_{31}&c_{32}&c_{33} \end{pmatrix}\).
Pour expliquer l'utilisation du masque écrivons le masque C
ainsi
qu'un voisinage d'un pixel im[i][j]
donné côte à côte
Si une image est représentée par la matrice \(M=(m_{ij})\) alors l'image \(M\oplus{}C=(m'_{ij})\) après avoir appliqué le masque de matrice \(C\) est (remarquer que ce n'est pas une multiplication matricielle)
\begin{equation*} m'_{ij} = \left\{\begin{array}{cccccccc} &c_{11}m_{(i-1);(j-1)}&&+&c_{12}m_{(i-1);j}&+&c_{13}m_{(i-1);(j+1)}\\ &+&c_{21}m_{i;(j-1)} &+&c_{22}m_{i;j} &+&c_{23}m_{i;(j+1)} \\ &+&c_{31}m_{(i+1);(j-1)}&+&c_{32}m_{(i+1);j}&+&c_{33}m_{(i+1);(j+1)} \end{array}\right. \end{equation*}Pour gagner du temps nous choisirons de ne pas appliquer le masque aux pixels qui sont sur le bord de l'image.
- Quelle opération effectue la convolution de masque \(C= \begin{pmatrix} 1 & 0 & 0 \\ 0 & 0 & 0 \\ 0 & 0 & 0 \end{pmatrix}\)?
Écrire une fonction
applique_masque_3x3(imtab, masque)
qui prend en argument une imageimtab
de taille \(n\times{}p\) et un masquemasque
de taille \(3x3\) w et qui retourne l'image de taille \((n-2)\times{}(p-2)\) après l'application du masque. Vérifier qu'avec le masque \(\begin{pmatrix}[[-1, 0,-1],[ 0, 5, 0],[-1, 0, -1]]\end{pmatrix}\) vous obtenez l'image qui suitAppliquer les masques de taille \(3\times{}3\) appelés ridge detection de cette page wikipedia à l'image en noir et blanc du cri de Munch.
Écrire une fonction
applique_masque(imtab, masque)
qui prend en argument une imageimtab
de taille \(n\times{}p\) et un masquemasque
de taille \(qxq\) où \(q=2k+1\) est impair et qui retourne l'image de taille \((n-k)\times{}(p-k)\) (pour éviter les problèmes aux bords…) après l'application du masque. On vérifiera que pour les masques[ [1/q**2]*q ]*q
pourq=11
etq=21
on obtient
3. Redimensionnement d'images
Ici on utilisera cette photo du chateau de saint Fargeaux que l'on
enregistrera encore dans un dossier img
du dossier courant.
import os img_st_farg_small_fpath = os.path.join("img", "saint_fargeaux_NB_small.png") img_st_farg_small = image_to_array(img_st_farg_small_fpath) plot_as_gray_image(img_st_farg_small)
Vérifier que le code ci-dessous affiche bien l'image qui lui suit
import os img_st_farg_small_fpath = os.path.join("img", "saint_fargeaux_NB_small.png") img_st_farg_small = image_to_array(img_st_farg_small_fpath) plot_as_gray_image(img_st_farg_small)
À cette question on souhaite redimensionner horizontalement une image à un nombre donné de pixels (la dimension verticale ne changeant pas pour simplifier). Pour cela
- pour chaque pixel \(P_{a}\) de l'image agrandie on détermine
les coordonnées d'un pixel \(P_{f}\) correspondant fictif car à
coordonnées flottantes dans l'image de départ. Ainsi par
exemple
- au pixel de coordonnées \((i,q)\) correspond celui de coordonnées \((i,p)\).
- si \(q=2r\) est pair alors au pixel de coordonnées \((n,r)\) correspond celui de coordonnées \((n,p/2)\) qui ne sont plus entières en général.
- on détermine une valeur naturel pour la couleur de \(P_{f}\) (par exemple couleur d'un pixel vrai càd à coordonnées entière le plus près ou autre)
- on affecte la couleur choisie pour \(P_{f}\) à \(P_{a}\).
On propose dans les questions qui suivent deux choix d'intensité pour le pixel fictif: dans un premier temps on lui affecte l'intensité du pixel le plus proche à sa gauche ou à sa droite puis dans un second temps on lui affecte une combinaison linéaire des intensités de ces deux pixels suivant leur distance à ces deux pixels adjacents.
- pour chaque pixel \(P_{a}\) de l'image agrandie on détermine
les coordonnées d'un pixel \(P_{f}\) correspondant fictif car à
coordonnées flottantes dans l'image de départ. Ainsi par
exemple
- Ici on choisit d'affecter l'intensité du pixel le plus près à
gauche ou à droite (nearest neighbour).
Compléter le code suivant
from math import floor def pixel_iind_jfrac_plus_pres(imtab, iind, jfrac): """Retourne le pixel situé à la ligne d'indice iind et à la colonne située au plus près de la colonne de position fractionnaire 0<=jfrac<=1 (jfrac=0 pour la 1ere colonne et jfrac=1 pour la dernière) imtab: tableau à 2 dimensions représentant une image N&B. iind: indice ligne valide de imtab. jfrac: position fractionnaire de la colonne où 0<=jfrac<=1 """ n, p = ?? i_res = iind # (j_virt=0 ssi jfrac=0) et (j_virt=p-1 ssi jfrac=1) j_virt = jfrac * (p-1) j_g = floor(j_virt) j_d = j_g + 1 if j_d > p - 1: assert j_g == p-1 return ?? pix_g, pix_d = imtab[i_res][j_g], imtab[i_res][j_d] if j_virt - j_g > ??: return ?? return ?? def test_pixel_iind_jfrac_plus_pres(): imtab = [[1, 2, 4], [0, 3, 7]] iind, jfrac = 0, 0.4 res = ?? assert pixel_iind_jfrac_plus_pres(imtab, iind, jfrac) == res imtab = [[1, 2, 4], [0, 10, 400]] iind, jfrac = 1, 0.2 res = ?? assert pixel_iind_jfrac_plus_pres(imtab, iind, jfrac) == res test_pixel_iind_jfrac_plus_pres()
Compléter le code suivant puis vérifier les résultat des deux blocs de code suivants (agrandissement d'une petite image et rétrécissement d'une grande)
import os from math import floor def resize_image_plus_pres(imtab, new_p): n = len(imtab) new_n = n imtab_resized = np.zeros((n, new_p)) for i in range(??): for j in range(??): jfrac = ?? / (??-1) new_pix = pixel_iind_jfrac_plus_pres(??, ??, ??) imtab_resized[??][??] = ?? return ??
img_st_farg_small_fpath = os.path.join("img", "saint_fargeaux_NB_small.png") img_st_farg_small = image_to_array(img_st_farg_small_fpath) img_st_farg_small_reszd = resize_image_plus_pres(img_st_farg_small, 400) plot_as_gray_image(img_st_farg_small_reszd)
img_st_farg_big_fpath = os.path.join("img", "saint_fargeaux_NB_big.png") img_st_farg_big = image_to_array(img_st_farg_big_fpath) img_st_farg_big_reszd = resize_image_plus_pres(img_st_farg_big, 100) plot_as_gray_image(img_st_farg_big_reszd)
- Ici on choisit d'affecter d'effectuer une interpolation affine
des intensités des deux pixels adjacents (situés à gauche et à
droite)
Compléter le code suivant
from math import floor def pixel_iind_jfrac_lin_interp(imtab, iind, jfrac): """Retourne le pixel situé à la ligne d'indice iind et à la colonne située au plus près de la colonne de position fractionnaire 0<=jfrac<=1 (jfrac=0 pour la 1ere colonne et jfrac=1 pour la dernière) imtab: tableau à 2 dimensions représentant une image N&B. iind: indice ligne valide de imtab. jfrac: position fractionnaire de la colonne où 0<=jfrac<=1 """ n, p = ?? i_res = iind # (j_virt=0 ssi jfrac=0) et (j_virt=p-1 ssi jfrac=1) j_virt = jfrac * (p-1) j_g = floor(j_virt) j_d = j_g + 1 if j_d > p - 1: assert j_g == p-1 return ?? pix_g, pix_d = imtab[i_res][j_g], imtab[i_res][j_d] pds_pix_d = j_virt - j_g pds_pix_g = 1 - pds_pix_d return ?? * pix_g + ?? * pix_d def aprx_eq(float1, float2): return abs(float1-float2) < 1.e-10 def test_aprx_eq(): assert aprx_eq(1, 1+1.e-11) assert aprx_eq(1+1.e-11, 1) assert aprx_eq(1, 1-1.e-11) assert aprx_eq(1-1.e-11, 1) assert not aprx_eq(1, 1+1.e-9) assert not aprx_eq(1+1.e-9, 1) assert not aprx_eq(1, 1-1.e-9) assert not aprx_eq(1-1.e-9, 1) test_aprx_eq() def test_pixel_iind_jfrac_lin_interp(): imtab = [[1, 2, 4], [0, 3, 7]] iind, jfrac = 0, 0.25 res = ?? assert aprx_eq(pixel_iind_jfrac_lin_interp(imtab, iind, jfrac), res) imtab = [[1, 2, 4], [0, 10, 400]] iind, jfrac = 1, 0.75 res = ?? assert aprx_eq(pixel_iind_jfrac_lin_interp(imtab, iind, jfrac), res) test_pixel_iind_jfrac_lin_interp()
Compléter le code suivant puis vérifier les résultat des deux blocs de code suivants (agrandissement d'une petite image et rétrécissement d'une grande)
import os from math import floor def resize_image_lin_interp(imtab, new_p): n = len(imtab) new_n = n imtab_resized = np.zeros((n, new_p)) for i in range(??): for j in range(??): jfrac = ?? / (??-1) new_pix = pixel_iind_jfrac_lin_interp(??, ??, ??) imtab_resized[??][??] = ?? return ??
img_st_farg_small_fpath = os.path.join("img", "saint_fargeaux_NB_small.png") img_st_farg_small = image_to_array(img_st_farg_small_fpath) img_st_farg_small_reszd = resize_image_lin_interp(img_st_farg_small, 400) plot_as_gray_image(img_st_farg_small_reszd)
img_st_farg_big_fpath = os.path.join("img", "saint_fargeaux_NB_big.png") img_st_farg_big = image_to_array(img_st_farg_big_fpath) img_st_farg_big_reszd = resize_image_lin_interp(img_st_farg_big, 100) plot_as_gray_image(img_st_farg_big_reszd)