Nommeur de photos

Soyez le premier à donner votre avis sur cette source.

Vue 5 049 fois - Téléchargée 267 fois

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 / Exemple :


# -*- 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)

Codes Sources

A voir également

Ajouter un commentaire

Commentaires

Messages postés
124
Date d'inscription
dimanche 5 octobre 2003
Statut
Membre
Dernière intervention
11 mai 2009
13
Merci!
J'avoue que j'ai un peu baclé la gestion des erreurs, rien n'apparait sur l'interface. Il faudrait que je remédie à ça.
Messages postés
382
Date d'inscription
mercredi 23 août 2006
Statut
Membre
Dernière intervention
8 novembre 2010
11
J'ai oublier la note !! Sorry
Messages postés
382
Date d'inscription
mercredi 23 août 2006
Statut
Membre
Dernière intervention
8 novembre 2010
11
Bien, très bien même. Le consepte est bon, l'interface est correcte, c'est du bon travail, sérieux qui fait plaisir à voir (il y en a trop peu). Je pense avoir repété quelque bugs en cas de mauvaise saisi de l'utilisateur, mais bon, ce n'est pas si grave ... Je pense que un petit 10 serais une note raisonnable, continu à nous faire de belles sources comme celle ci !
____
Aéra

Vous n'êtes pas encore membre ?

inscrivez-vous, c'est gratuit et ça prend moins d'une minute !

Les membres obtiennent plus de réponses que les utilisateurs anonymes.

Le fait d'être membre vous permet d'avoir un suivi détaillé de vos demandes et codes sources.

Le fait d'être membre vous permet d'avoir des options supplémentaires.