begin process at 2008 05 12 10:49:41
1 170 205 membres
108 nouveaux aujourd'hui
13 957 membres club

Vous ne trouvez pas de réponse à votre problème ? Alors posez la question dans le forum.
Souvenez-vous qu'il n'y a jamais de question bête, mais rester dans l'ignorance parce que l'on n'ose pas poser une question, ça c'est une erreur !

[PYGTK] LECTEUR MULTI-VIDÉO EMBARQUANT MPLAYER GRÂCE À L'ID WINDOW D'UNE SOCKET.


Information sur la source

Catégorie :Graphique Classé sous : pygtk, mplayer, python, socket, video Niveau : Initié Date de création : 01/04/2008 Date de mise à jour : 01/04/2008 12:10:43 Vu : 990

Note :
Aucune note

Commentaire sur cette source (0)
Ajouter un commentaire et/ou une note

Description

Ce lecteur multi-vidéo embarque Mplayer, à travers l'utilisation de la classe GtkSocket de Pygtk, qui permet d'intégrer n'importe quelle interface, grâce à son identification(id Window).
Écrit en python, ce lecteur permet de lire jusqu'à 4 vidéos simultanément.
Mplayer est utilisé en mode slave(cf documentation mplayer) pour permettre un contrôle par commande au travers d'un tube. A noter également son option '-wid' qui spécifie l'id sur laquelle il s'accrochera.

Globalement il y a 3classes. La première est l'interface avec mplayer, elle crée le tube, et propose quelque commande.
La seconde définit un player complet(vidéo, + widget de controle : play, rewind, forward, ouvrir un fichier) et utilise la 1ere classe pour interargir avec les widgets.

La troisième et dernière classe, est la classe de base de l'application. Elle s'occupe des routines habituelles avec pygtk, et crée les 4 players(En instanciant donc 4fois la seconde classe).
  
Pourquoi ?
Parceque tout simplement j'ai essayé une tonne de librairie pour "manipuler" de la vidéo dans mon application, ce qui fut un échec total.
Notamment pymedia, salement documenté(des exemples très peu explicite, une gui utilisant pygame...etc). Ou pygst, qui dispose d'une bonne documentation(pas complète mais interressante), mais pas stable lorsque j'ai essayé différents formats, ou lorsqu'on essaye de se déplacer dans la vidéo, le programme plante...

La meilleure solution était donc d'intégrer directement un player et d'intérargir avec. D'autant plus que si vous cliquez sur la vidéo, vous pouvez utiliser les touches supportés par mplayer!(Exemple : 'p' pour pause...etc).

En espérant que cette modeste contribution aide certains d'entres-vous!

Environnement de développement :
x86, Ubuntu, noyau 2.6
Python 2.5
Pygtk 2.0

Source

  • # -*- Encoding: Latin-1 -*-
  • import pygtk
  • pygtk.require('2.0')
  • import gtk
  • import os
  • import psyco
  • psyco.full()
  • twindow = gtk.EXPAND | gtk.FILL | gtk.SHRINK
  • PATH_mkfifo = '/tmp/mplayer'
  • class Pymplayer(gtk.Socket):
  • """Classe d'interface entre mplayer et le programme"""
  • def __init__(self, id):
  • gtk.Socket.__init__(self)
  • self.tube = PATH_mkfifo + str(id)
  • try:
  • #Supprimer le fichier tube si présent.
  • os.unlink(self.tube)
  • except:
  • pass
  • #On crée le tube, son nom est mplayer+id
  • os.mkfifo(self.tube)
  • os.chmod(self.tube, 0666)
  • #On écrit dans le tube la commande en mode slave
  • def execmplayer(self, cmd):
  • open(self.tube,'w').write(cmd+"\n")
  • #On lance mplayer en lui précisant l'id de la widget à utiliser(cf mplayer doc pour le reste)
  • def setwid(self, wid):
  • os.system("mplayer -nojoystick -nolirc -slave -vo x11 -wid %s -vf scale=400:200 -idle -input file=%s &"%(wid, self.tube))
  • def loadfile(self, filename):
  • self.execmplayer("loadfile %s"%(filename))
  • self.execmplayer("change_rectangle w=100:h=100")
  • def pause(self, parent):
  • self.execmplayer("pause")
  • def forward(self, parent):
  • self.execmplayer("seek +10 0")
  • def backward(self, parent):
  • self.execmplayer("seek -10 0")
  • def quit(self):
  • self.execmplayer("quit")
  • class Panel(gtk.Table):
  • """ Elle definie le player, càd l'écran, et ses widgets(play, forward, backward...)"""
  • def __init__(self, id):
  • gtk.Table.__init__(self, 4, 2)
  • #self.resize(4, 2)
  • self.set_col_spacings(10)
  • self.set_row_spacings(10)
  • self.Ecran = Pymplayer(id)
  • self.Ecran.set_size_request(100, 200)
  • self.attach(self.Ecran, 0, 4, 0, 1, twindow, twindow, 5, 5)
  • #Bouton Open file
  • Bopenfile = gtk.Button(stock="gtk-open")
  • Bopenfile.connect("clicked", self.openfile)
  • self.attach(Bopenfile, 0, 1, 1, 2, twindow, twindow, 5, 5)
  • #Bouton previous
  • Brewind = gtk.Button(stock="gtk-media-rewind")
  • Brewind.connect("clicked", self.Ecran.backward)
  • self.attach(Brewind, 1, 2, 1, 2, twindow, twindow, 5, 5)
  • #Bouton play
  • Bplay = gtk.Button(stock="gtk-media-play")
  • Bplay.connect("clicked", self.Ecran.pause)
  • self.attach(Bplay, 2, 3, 1, 2, twindow, twindow, 5, 5)
  • #Bouton next
  • Bforward = gtk.Button(stock="gtk-media-forward")
  • Bforward.connect("clicked", self.Ecran.forward)
  • self.attach(Bforward, 3, 4, 1, 2, twindow, twindow, 5, 5)
  • #Boite de dialogue qui permet à l'utilisateur de choisir un fichire
  • def openfile(self, parent):
  • dialog = gtk.FileChooserDialog("Select file", gtk.Window(), gtk.FILE_CHOOSER_ACTION_OPEN,
  • (gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
  • gtk.STOCK_OPEN, gtk.RESPONSE_OK))
  • dialog.connect("destroy",lambda w:dialog.destroy())
  • dialog.set_default_response(gtk.RESPONSE_OK)
  • statut = dialog.run()
  • if statut == gtk.RESPONSE_OK:
  • #Ok nous avons le fichier, ouvrons le avec mplayer!
  • self.Ecran.loadfile(dialog.get_filename())
  • dialog.destroy()
  • class guiPymplayer:
  • """ Classe maitre, qui inclut toutes les autres classes"""
  • def __init__(self):
  • self.gui = gtk.Window(gtk.WINDOW_TOPLEVEL)
  • self.gui.set_title("Using plug and socket : mplayer in python, by loupmagic")
  • self.gui.set_default_size(550, 400)
  • self.gui.set_position(gtk.WIN_POS_CENTER)
  • self.global_panel()
  • self.gui.connect("destroy", self.do_quit)
  • self.gui.show_all()
  • #Envoie des id seulement après l'affichage des widgets à l'aide de show_all()
  • self.Panel1.Ecran.setwid(long(self.Panel1.Ecran.get_id()))
  • self.Panel2.Ecran.setwid(long(self.Panel2.Ecran.get_id()))
  • self.Panel3.Ecran.setwid(long(self.Panel3.Ecran.get_id()))
  • self.Panel4.Ecran.setwid(long(self.Panel4.Ecran.get_id()))
  • def global_panel(self):
  • #Table d'organisation des widgets
  • table = gtk.Table(3, 3, False)
  • self.Panel1 = Panel(1)
  • table.attach(self.Panel1, 0, 1, 0, 1, twindow, twindow, 5, 5)
  • self.Panel2 = Panel(2)
  • table.attach(self.Panel2, 2, 3, 0, 1, twindow, twindow, 5, 5)
  • self.Panel3 = Panel(3)
  • table.attach(self.Panel3, 0, 1, 2, 3, twindow, twindow, 5, 5)
  • self.Panel4 = Panel(4)
  • table.attach(self.Panel4, 2, 3, 2, 3, twindow, twindow, 5, 5)
  • ligneH = gtk.HSeparator()
  • table.attach(ligneH, 0, 3, 1, 2)
  • ligneV = gtk.VSeparator()
  • table.attach(ligneV, 1, 2, 0, 3)
  • self.gui.add(table)
  • def do_quit(self, widget):
  • if __name__ == '__main__':
  • #Fermer les processus mplayer!
  • self.Panel1.Ecran.quit()
  • self.Panel2.Ecran.quit()
  • self.Panel3.Ecran.quit()
  • self.Panel4.Ecran.quit()
  • gtk.main_quit()
  • def loop(self):
  • gtk.main()
  • multivideos = guiPymplayer()
  • multivideos.loop()
# -*- Encoding: Latin-1 -*-

import pygtk
pygtk.require('2.0')
import gtk
import os 
import psyco
psyco.full()

twindow = gtk.EXPAND | gtk.FILL | gtk.SHRINK

PATH_mkfifo = '/tmp/mplayer'

class Pymplayer(gtk.Socket):
	"""Classe d'interface entre mplayer et le programme"""
	def __init__(self, id):
		gtk.Socket.__init__(self)
		self.tube = PATH_mkfifo + str(id)		
		try:
			#Supprimer le fichier tube si présent.
			os.unlink(self.tube)
		except:
			pass
		#On crée le tube, son nom est mplayer+id
		os.mkfifo(self.tube)
		os.chmod(self.tube, 0666)

	#On écrit dans le tube la commande en mode slave
	def execmplayer(self, cmd):	
		open(self.tube,'w').write(cmd+"\n")
	
	#On lance mplayer en lui précisant l'id de la widget à utiliser(cf mplayer doc pour le reste)
	def setwid(self, wid):
		os.system("mplayer -nojoystick -nolirc -slave -vo x11 -wid %s -vf scale=400:200 -idle -input file=%s &"%(wid, self.tube))

	def loadfile(self, filename):
		self.execmplayer("loadfile %s"%(filename))
		self.execmplayer("change_rectangle w=100:h=100")

	def pause(self, parent):
		self.execmplayer("pause")

	def forward(self, parent):
		self.execmplayer("seek +10 0")

	def backward(self, parent):
		self.execmplayer("seek -10 0")		

	def quit(self):
		self.execmplayer("quit")



class Panel(gtk.Table):
	""" Elle definie le player, càd l'écran, et ses widgets(play, forward, backward...)"""

	def __init__(self, id):
		gtk.Table.__init__(self, 4, 2)
		#self.resize(4, 2)
		self.set_col_spacings(10)
	        self.set_row_spacings(10)
		
		self.Ecran = Pymplayer(id)
  		self.Ecran.set_size_request(100, 200)
		self.attach(self.Ecran, 0, 4, 0, 1, twindow, twindow, 5, 5)

		#Bouton Open file
		Bopenfile = gtk.Button(stock="gtk-open")
		Bopenfile.connect("clicked", self.openfile)
		self.attach(Bopenfile, 0, 1, 1, 2, twindow, twindow, 5, 5)
	
		#Bouton previous 		
		Brewind = gtk.Button(stock="gtk-media-rewind")
		Brewind.connect("clicked", self.Ecran.backward)
		self.attach(Brewind, 1, 2, 1, 2, twindow, twindow, 5, 5)
	
		#Bouton play
		Bplay = gtk.Button(stock="gtk-media-play")
		Bplay.connect("clicked", self.Ecran.pause)
		self.attach(Bplay, 2, 3, 1, 2, twindow, twindow, 5, 5)

		#Bouton next 		
		Bforward = gtk.Button(stock="gtk-media-forward")
		Bforward.connect("clicked", self.Ecran.forward)
		self.attach(Bforward, 3, 4, 1, 2, twindow, twindow, 5, 5)


	#Boite de dialogue qui permet à l'utilisateur de choisir un fichire
	def openfile(self, parent):
  		dialog = gtk.FileChooserDialog("Select file", gtk.Window(), gtk.FILE_CHOOSER_ACTION_OPEN,
                                    (gtk.STOCK_CANCEL, gtk.RESPONSE_CANCEL,
                                    gtk.STOCK_OPEN, gtk.RESPONSE_OK))
		dialog.connect("destroy",lambda w:dialog.destroy())
    	        dialog.set_default_response(gtk.RESPONSE_OK)

		statut = dialog.run()
    	        if statut == gtk.RESPONSE_OK:
			#Ok nous avons le fichier, ouvrons le avec mplayer!
       	 		self.Ecran.loadfile(dialog.get_filename())
    		dialog.destroy()


class guiPymplayer:
	""" Classe maitre, qui inclut toutes les autres classes"""

	def __init__(self):	
		self.gui = gtk.Window(gtk.WINDOW_TOPLEVEL)
		self.gui.set_title("Using plug and socket : mplayer in python, by loupmagic")
		self.gui.set_default_size(550, 400)
		self.gui.set_position(gtk.WIN_POS_CENTER)
		
		self.global_panel()

		self.gui.connect("destroy", self.do_quit)
		self.gui.show_all()
		
		#Envoie des id seulement après l'affichage des widgets à l'aide de show_all()
		self.Panel1.Ecran.setwid(long(self.Panel1.Ecran.get_id()))	
		self.Panel2.Ecran.setwid(long(self.Panel2.Ecran.get_id()))
		self.Panel3.Ecran.setwid(long(self.Panel3.Ecran.get_id()))
		self.Panel4.Ecran.setwid(long(self.Panel4.Ecran.get_id()))


	def global_panel(self):
		#Table d'organisation des widgets
		table = gtk.Table(3, 3, False)

		self.Panel1 = Panel(1)
		table.attach(self.Panel1, 0, 1, 0, 1, twindow, twindow, 5, 5)
		
		self.Panel2 = Panel(2)
		table.attach(self.Panel2, 2, 3, 0, 1, twindow, twindow, 5, 5)

		self.Panel3 = Panel(3)
		table.attach(self.Panel3, 0, 1, 2, 3, twindow, twindow, 5, 5)

		self.Panel4 = Panel(4)
		table.attach(self.Panel4, 2, 3, 2, 3, twindow, twindow, 5, 5)

		ligneH = gtk.HSeparator()
		table.attach(ligneH, 0, 3, 1, 2)
		ligneV = gtk.VSeparator()
		table.attach(ligneV, 1, 2, 0, 3)

		self.gui.add(table)

	def do_quit(self, widget):
		if __name__ == '__main__':
			#Fermer les processus mplayer!
			self.Panel1.Ecran.quit()
			self.Panel2.Ecran.quit()
			self.Panel3.Ecran.quit()
			self.Panel4.Ecran.quit()
		        gtk.main_quit()

	def loop(self):
		gtk.main()

multivideos = guiPymplayer()
multivideos.loop()

Conclusion

Le plus long a été la recherche consacré aux différents modules vidéos sous python, avec pygtk, puisque ce programme a été développé en 2H30(j'ai déjà réalisé une autre application, un peu plus précise, sur le même style).

En effet, la documentation francaise est quasi-inexistante, même aux niveaux des exemples.
En anglais, un peu mieux, mais toujours pas d'élément concret démontrant l'utilité de telle ou telle classe sous python.

Cette solution me semble la plus approprié si vous voulez simplement embarqué un programme.

Le code est abordable. Il utilise simplement des classes hérités pour certaine de widget spécifique.
Il quitte en nettoyant les processus lancés au démarrage.

01 avril 2008 12:08:30 :
Ajout de l'environnement de développement (OS, version...etc)
01 avril 2008 12:10:43 :
Ajout de l'environnement de développement(OS, version...etc)
    Aucun commentaire pour le moment.

Ajouter un commentaire

Discussions en rapport avec ce code source

tutoriaux python par slachz

installation python par champagnef

Projet Python: Lecteur de flux RSS. par Marvin

Paramétrage de python par deguelatore

Tk vs wx par bonac

Help : Python/DB/CGI par Chill_Sik

WebCam par DoudouBidou

dll avec python? par LokR

aide pour xchat svp par Jamu

les variables vu par Plone... par onlybjork

Appels d'offres

Pub



CalendriCode

Mai 2008
LMMJVSD
   1234
567891011
12131415161718
19202122232425
262728293031 

Téléchargements

Boutique

Boutique de goodies CodeS-SourceS