Accueil > > > NOMMEUR DE PHOTOS
NOMMEUR DE PHOTOS
Information sur la source
Description
Voici une interface graphique simple (en Tk), permettant de nommer les photos se trouvant dans un répertoire donné, de la manière la plus simple possible. En effet, j'ai constaté que lorsque l'on prend des photos avec un appareil numérique, on se retrouve avec un tas de photos aux noms peu aguicheurs. Lorsqu'on a une série de 50 photos à renommer, quelquesoit le système d'exploitation, il n'existe pas de solution simple. Ce "nommeur" affiche la photo, vous entrez son nom, appuyez sur entrée, la photo suivante apparait etc... Pour moi, c'est bien utile. Nécessite PIL
Source
- # -*- coding: cp1252 -*-
- import sys, os, Image, ImageTk, Tkinter, gc, time, tkFileDialog, tkMessageBox
- class IHM_Nommeur(Tkinter.Frame):
-
- #retourne l'image redimensionnée en gardant les proportions
- def resize(self, im, x, y):
- if (float(im.size[0])/float(x)>float(im.size[1])/float(y)):
- coeffResize=float(im.size[0])/float(x)
- else:
- coeffResize=float(im.size[1])/float(y)
- return im.resize((int(round((im.size[0]/coeffResize))),int(round((im.size[1]/coeffResize)))))
-
- # constructeur
- def __init__(self, title="VyCHNou'S piCTuRe^NaMeR", master=None):
- Tkinter.Frame.__init__(self, master)
- self.createWidgets()
- self.master.title(title)
- self.nom_app=title
-
- #executée au chargement d'un dossier
- def start(self, dossier):
- self.num_photo=-1
- self.nom_photo=None
- self.focus_force()
- self.e_denomination.focus_set()
- self.dossier=dossier+os.sep
- self.listdir=[]
- for f in os.listdir(self.dossier):
- try:
- im=Image.open(self.dossier+f, "r")
- self.listdir.append(f)
- except:
- None
- if (len(self.listdir)==0):
- tkMessageBox.showwarning('Erreur',"Pas de photo dans le répertoire selectionné.")
- self.next()
-
- # initialisation des widgets
- def createWidgets(self):
- self.font=("ms san serif", "16")
- # Création des widgets esclaves(dans l'ordre du chemin focus) :
- commandes=Tkinter.Frame(self.master)
-
- self.b_previous=Tkinter.Button(commandes, text="<=", command=self.previous, font=self.font)
- self.e_denomination= Tkinter.Entry(commandes, width=50, font=self.font)
- self.e_extension=Tkinter.Entry(commandes, width=7,font=self.font)
- self.b_next=Tkinter.Button(commandes, text="=>", command=self.next, font=self.font)
- self.b_load=Tkinter.Button(commandes, text="ouvrir", command=self.load, font=self.font)
- self.b_help=Tkinter.Button(commandes, text="?", command=self.help, font=self.font)
- self.c_afficheur = Tkinter.Canvas(height=800,width=800)
- self.cv_sens=Tkinter.IntVar()
- self.cv_sens.set(2)
- self.rb_previous=Tkinter.Radiobutton(commandes, variable=self.cv_sens, value=1)
- self.rb_next=Tkinter.Radiobutton(commandes, variable=self.cv_sens, value=2)
- l_denomination=Tkinter.Label(commandes, text="dénomination")
- l_extension=Tkinter.Label(commandes, text="extension")
-
- #placement des widgets
- self.e_denomination.grid(row=1, column=1)
- l_denomination.grid(row=0, column=1)
- self.e_extension.grid(row=1, column=2)
- l_extension.grid(row=0, column=2)
- self.b_previous.grid(row=1, column=0)
- self.b_next.grid(row=1, column=3)
- self.b_load.grid(row=1, column=4)
- self.b_help.grid(row=1, column=5)
- self.rb_previous.grid(row=2, column=0)
- self.rb_next.grid(row=2, column=3)
- commandes.pack({"side":"bottom","expand":"no","anchor":"s"})
- self.c_afficheur.pack({"side":"top","expand":"yes","fill":"both","anchor":"center","padx":"5","pady":"5"})
-
- self.e_denomination.focus_set()
- self.pack()
-
- # définition des binds
- self.e_denomination.bind("<Return>", self.nomme)
- self.e_denomination.bind("<Escape>", self.quit)
- self.e_denomination.bind("<Prior>", self.previous)
- self.e_denomination.bind("<Next>", self.next)
- self.e_denomination.bind("<Control-Key-o>", self.load)
- self.e_denomination.bind("<F1>", self.help)
- self.e_extension.bind("<Return>", self.nomme)
- self.e_extension.bind("<Escape>", self.quit)
- self.e_extension.bind("<Prior>", self.previous)
- self.e_extension.bind("<Next>", self.next)
- self.e_extension.bind("<Control-Key-o>", self.load)
- self.e_extension.bind("<F1>", self.help)
-
- # nomme le fichier
- def nomme(self, event):
- if(self.nom_photo==None):
- return
- self.c_afficheur.delete(Tkinter.ALL)
- if (self.nom_photo!=self.e_denomination.get()+self.e_extension.get()):
- os.rename(self.dossier+self.nom_photo, self.dossier+self.e_denomination.get()+self.e_extension.get())
- # on renomme aussi dans la liste des fichiers
- self.listdir[self.num_photo]=self.e_denomination.get()+self.e_extension.get()
- if(self.cv_sens.get()==1):
- self.previous()
- else:
- self.next()
-
- #passe à la photo suivante
- def next(self, event=0):
- self.num_photo=self.num_photo+1
- if (self.num_photo>=len(self.listdir)):
- self.num_photo=0
- self.actualise_affichage()
-
- #passe à la photo précédente
- def previous(self, event=0):
- self.num_photo=self.num_photo-1
- if (self.num_photo<0):
- self.num_photo=len(self.listdir)-1
- self.actualise_affichage()
-
- #après un changement de photo
- def actualise_affichage(self):
- if len(self.listdir)<=0:
- return
- self.nom_photo=self.listdir[self.num_photo]
- try:
- im=Image.open(self.dossier+self.nom_photo, "r")
- except:
- self.next()
- return()
- self.nom_photo=self.listdir[self.num_photo]
- im=Image.open(self.dossier+self.nom_photo, "r")
- self.e_denomination.delete(0, Tkinter.END)
- locPoint=self.nom_photo.rfind(".")
- if(locPoint==-1):
- locPoint=len(self.nom_photo)
- self.e_denomination.insert(0, self.nom_photo[0: locPoint])
- self.e_denomination.selection_range(0,Tkinter.END)
- self.e_extension.delete(0, Tkinter.END)
- self.e_extension.insert(0, self.nom_photo[locPoint: len(self.nom_photo)])
- im2=self.resize(im, int(self.c_afficheur.winfo_width()), int(self.c_afficheur.winfo_height()))
- imTk=ImageTk.PhotoImage(im2)
- self.c_afficheur.create_image(0,0,image=imTk,anchor='nw')
- self.c_afficheur.img=imTk
- self.master.title(self.nom_app+" - "+self.dossier+" - photo "+str(self.num_photo+1)+"/"+str(len(self.listdir)))
- self.pack()
- self.c_afficheur.update_idletasks()
-
- # affiche l'aide
- def help(self, event=0):
- helpMsg="""Ce programme a pour but d'aider au nommage des fichiers photos.
- Il existe des raccourcis claviers:
- -Entrée: nomme la photo et passe à la suivante ou à la précédente en fonction du bouton radio selectionné
- -Page up: photo précédente
- -Page down: photo suivante
- -Ctrl+O: ouvrir un autre dossier
- -F1: affiche cette aide
- -Echap: Quitte le programme
-
- Ecrit par VyCHNou, pour toutes questions:
- vychnou@hotmail.com"""
- tkMessageBox.showinfo(title="aide", message=helpMsg)
-
- # quitte
- def quit(self, event=0):
- sys.exit(0)
-
- #permet de changer le dossier
- def load(self, event=0):
- file=tkFileDialog.askdirectory(initialdir="/", title="Choisissez le dossier ou se trouvent les photos à renommer",mustexist="true")
- fen.start(file)
-
- fen=IHM_Nommeur()
- fen.mainloop()#let's run the window
# -*- coding: cp1252 -*-
import sys, os, Image, ImageTk, Tkinter, gc, time, tkFileDialog, tkMessageBox
class IHM_Nommeur(Tkinter.Frame):
#retourne l'image redimensionnée en gardant les proportions
def resize(self, im, x, y):
if (float(im.size[0])/float(x)>float(im.size[1])/float(y)):
coeffResize=float(im.size[0])/float(x)
else:
coeffResize=float(im.size[1])/float(y)
return im.resize((int(round((im.size[0]/coeffResize))),int(round((im.size[1]/coeffResize)))))
# constructeur
def __init__(self, title="VyCHNou'S piCTuRe^NaMeR", master=None):
Tkinter.Frame.__init__(self, master)
self.createWidgets()
self.master.title(title)
self.nom_app=title
#executée au chargement d'un dossier
def start(self, dossier):
self.num_photo=-1
self.nom_photo=None
self.focus_force()
self.e_denomination.focus_set()
self.dossier=dossier+os.sep
self.listdir=[]
for f in os.listdir(self.dossier):
try:
im=Image.open(self.dossier+f, "r")
self.listdir.append(f)
except:
None
if (len(self.listdir)==0):
tkMessageBox.showwarning('Erreur',"Pas de photo dans le répertoire selectionné.")
self.next()
# initialisation des widgets
def createWidgets(self):
self.font=("ms san serif", "16")
# Création des widgets esclaves(dans l'ordre du chemin focus) :
commandes=Tkinter.Frame(self.master)
self.b_previous=Tkinter.Button(commandes, text="<=", command=self.previous, font=self.font)
self.e_denomination= Tkinter.Entry(commandes, width=50, font=self.font)
self.e_extension=Tkinter.Entry(commandes, width=7,font=self.font)
self.b_next=Tkinter.Button(commandes, text="=>", command=self.next, font=self.font)
self.b_load=Tkinter.Button(commandes, text="ouvrir", command=self.load, font=self.font)
self.b_help=Tkinter.Button(commandes, text="?", command=self.help, font=self.font)
self.c_afficheur = Tkinter.Canvas(height=800,width=800)
self.cv_sens=Tkinter.IntVar()
self.cv_sens.set(2)
self.rb_previous=Tkinter.Radiobutton(commandes, variable=self.cv_sens, value=1)
self.rb_next=Tkinter.Radiobutton(commandes, variable=self.cv_sens, value=2)
l_denomination=Tkinter.Label(commandes, text="dénomination")
l_extension=Tkinter.Label(commandes, text="extension")
#placement des widgets
self.e_denomination.grid(row=1, column=1)
l_denomination.grid(row=0, column=1)
self.e_extension.grid(row=1, column=2)
l_extension.grid(row=0, column=2)
self.b_previous.grid(row=1, column=0)
self.b_next.grid(row=1, column=3)
self.b_load.grid(row=1, column=4)
self.b_help.grid(row=1, column=5)
self.rb_previous.grid(row=2, column=0)
self.rb_next.grid(row=2, column=3)
commandes.pack({"side":"bottom","expand":"no","anchor":"s"})
self.c_afficheur.pack({"side":"top","expand":"yes","fill":"both","anchor":"center","padx":"5","pady":"5"})
self.e_denomination.focus_set()
self.pack()
# définition des binds
self.e_denomination.bind("<Return>", self.nomme)
self.e_denomination.bind("<Escape>", self.quit)
self.e_denomination.bind("<Prior>", self.previous)
self.e_denomination.bind("<Next>", self.next)
self.e_denomination.bind("<Control-Key-o>", self.load)
self.e_denomination.bind("<F1>", self.help)
self.e_extension.bind("<Return>", self.nomme)
self.e_extension.bind("<Escape>", self.quit)
self.e_extension.bind("<Prior>", self.previous)
self.e_extension.bind("<Next>", self.next)
self.e_extension.bind("<Control-Key-o>", self.load)
self.e_extension.bind("<F1>", self.help)
# nomme le fichier
def nomme(self, event):
if(self.nom_photo==None):
return
self.c_afficheur.delete(Tkinter.ALL)
if (self.nom_photo!=self.e_denomination.get()+self.e_extension.get()):
os.rename(self.dossier+self.nom_photo, self.dossier+self.e_denomination.get()+self.e_extension.get())
# on renomme aussi dans la liste des fichiers
self.listdir[self.num_photo]=self.e_denomination.get()+self.e_extension.get()
if(self.cv_sens.get()==1):
self.previous()
else:
self.next()
#passe à la photo suivante
def next(self, event=0):
self.num_photo=self.num_photo+1
if (self.num_photo>=len(self.listdir)):
self.num_photo=0
self.actualise_affichage()
#passe à la photo précédente
def previous(self, event=0):
self.num_photo=self.num_photo-1
if (self.num_photo<0):
self.num_photo=len(self.listdir)-1
self.actualise_affichage()
#après un changement de photo
def actualise_affichage(self):
if len(self.listdir)<=0:
return
self.nom_photo=self.listdir[self.num_photo]
try:
im=Image.open(self.dossier+self.nom_photo, "r")
except:
self.next()
return()
self.nom_photo=self.listdir[self.num_photo]
im=Image.open(self.dossier+self.nom_photo, "r")
self.e_denomination.delete(0, Tkinter.END)
locPoint=self.nom_photo.rfind(".")
if(locPoint==-1):
locPoint=len(self.nom_photo)
self.e_denomination.insert(0, self.nom_photo[0: locPoint])
self.e_denomination.selection_range(0,Tkinter.END)
self.e_extension.delete(0, Tkinter.END)
self.e_extension.insert(0, self.nom_photo[locPoint: len(self.nom_photo)])
im2=self.resize(im, int(self.c_afficheur.winfo_width()), int(self.c_afficheur.winfo_height()))
imTk=ImageTk.PhotoImage(im2)
self.c_afficheur.create_image(0,0,image=imTk,anchor='nw')
self.c_afficheur.img=imTk
self.master.title(self.nom_app+" - "+self.dossier+" - photo "+str(self.num_photo+1)+"/"+str(len(self.listdir)))
self.pack()
self.c_afficheur.update_idletasks()
# affiche l'aide
def help(self, event=0):
helpMsg="""Ce programme a pour but d'aider au nommage des fichiers photos.
Il existe des raccourcis claviers:
-Entrée: nomme la photo et passe à la suivante ou à la précédente en fonction du bouton radio selectionné
-Page up: photo précédente
-Page down: photo suivante
-Ctrl+O: ouvrir un autre dossier
-F1: affiche cette aide
-Echap: Quitte le programme
Ecrit par VyCHNou, pour toutes questions:
vychnou@hotmail.com"""
tkMessageBox.showinfo(title="aide", message=helpMsg)
# quitte
def quit(self, event=0):
sys.exit(0)
#permet de changer le dossier
def load(self, event=0):
file=tkFileDialog.askdirectory(initialdir="/", title="Choisissez le dossier ou se trouvent les photos à renommer",mustexist="true")
fen.start(file)
fen=IHM_Nommeur()
fen.mainloop()#let's run the window
Conclusion
J'ai prévu de faire une amélioration d'ici peu: proposer un suffixe automatique (exemple: vacances_a_valence, où alors heure_date de la photo)
Historique
- 26 septembre 2007 15:05:04 :
- correction d'un bug qui empechait le choix de la direction
Sources du même auteur
Sources de la même categorie
Commentaires et avis
|
Derniers Blogs
TECHDAYS PARIS 2010 : LA BI DANS SHAREPOINT 2010TECHDAYS PARIS 2010 : LA BI DANS SHAREPOINT 2010 par ROMELARD Fabrice
Animé par: Vincent Bellet et Baptiste Giraudier La BI dans SharePoint 2010, Les nouveaux services d'application dans SP2010 et SQL Server Reporting services 2008 R2. La BI dans SharePoint est généralisée pour tous afin de permettre à tous les coll...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice TECHDAYS PARIS 2010 : PLAN DE MIGRATION VERS SHAREPOINT 2010TECHDAYS PARIS 2010 : PLAN DE MIGRATION VERS SHAREPOINT 2010 par ROMELARD Fabrice
Animé par: Arnault Nouvel et Antoine Dongois Le processus à prendre : Apprendre (découvrir la plateforme) Préparer (documenter l'historique et choisir la méthode de MAJ) Test (Test de MAJ) Implémenter (Effectuer la MAJ) Valid...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice TECHDAYS PARIS 2010 : LA PLEINIèRE DU SECOND JOURTECHDAYS PARIS 2010 : LA PLEINIèRE DU SECOND JOUR par ROMELARD Fabrice
Après un retour sur l'histoire des TechDays de Paris et le fait que ce soit le plus gros event MS au monde (du fait de sa gratuité), le président de MS France (Eric Boustoullier) a fait une présentation de la vision Microsoft pour les années à venir...
Cliquez pour lire la suite de l'article par ROMELARD Fabrice
Logiciels
DB-MAIN (9.1.0)DB-MAIN (9.1.0)DB-MAIN is a data-modeling and data-architecture tool. It is designed to help developers and anal... Cliquez pour télécharger DB-MAIN Xilisoft DPG Convertisseur (5.1.37.0120)XILISOFT DPG CONVERTISSEUR (5.1.37.0120)Xilisoft DPG Convertisseur offre aux fans de Nintendo DS une bonne solution leur permettant de dé... Cliquez pour télécharger Xilisoft DPG Convertisseur GraphicsGale (2.01.01)GRAPHICSGALE (2.01.01)GraphicsGale est un logiciel de PixelArt avec de nombreuse fonctionnalités permettant de réalisé ... Cliquez pour télécharger GraphicsGale Architecte 3D (Platinum 2010)ARCHITECTE 3D (PLATINUM 2010)Architecte 3D Platinium vous permet de concevoir facilement les plans votre future maison, de l'é... Cliquez pour télécharger Architecte 3D TeamViewer 5 (TeamViewer 5)TEAMVIEWER 5 (TEAMVIEWER 5)Dépanner un ami,expliquer une manipulation devient un jeu d'enfant.
Prise en main d'un autre ord... Cliquez pour télécharger TeamViewer 5
|