mirror of
https://github.com/jlengrand/Ivolution.git
synced 2026-03-10 08:21:18 +00:00
374 lines
14 KiB
Python
374 lines
14 KiB
Python
#!/usr/bin/env python
|
|
"""
|
|
.. module:: IvolutionWindow
|
|
:platform: Unix, Windows, Mac
|
|
:synopsis: Main Window of the Ivolution GUI designed to be supported by all platforms.
|
|
|
|
.. moduleauthor:: Julien Lengrand-Lambert <jlengrand@gmail.com>
|
|
|
|
"""
|
|
|
|
import wx
|
|
import wx.lib.newevent
|
|
|
|
import sys
|
|
import os
|
|
import logging
|
|
import webbrowser
|
|
|
|
from .. import get_data # used to load images and files
|
|
|
|
from .. import FaceParams
|
|
from .. import FacemovieThread
|
|
|
|
from ..util.Notifier import Observer
|
|
from ..util.Notifier import Observable
|
|
|
|
from IvolutionTemplate import IvolutionTemplate
|
|
from SettingsWindow import SettingsWindow
|
|
|
|
|
|
class IvolutionWindow(IvolutionTemplate, Observer, Observable):
|
|
"""
|
|
Main Window of the Ivolution application
|
|
"""
|
|
def __init__(self, parent, title):
|
|
"""
|
|
Overrides init frame IvolutionTemplate
|
|
"""
|
|
IvolutionTemplate.__init__(self, parent)
|
|
Observer.__init__(self, "Interface")
|
|
Observable.__init__(self)
|
|
|
|
# Sets up logging capability
|
|
self.my_logger = None
|
|
self.console_logger = None
|
|
self.setup_logger()
|
|
|
|
# Defines all our parameters neededfor the facemovie
|
|
self.get_default_parameters()
|
|
|
|
self.process_running = False
|
|
self.facemovie = None
|
|
|
|
self.inputtextbox.SetLabel(self.in_fo) # sets label to default input folder
|
|
self.SetIcon(wx.Icon('ivolution/data/media/vitruve.ico', wx.BITMAP_TYPE_ICO)) # Sets icon
|
|
|
|
self.Show(True) # Finally show the frame
|
|
|
|
def get_default_parameters(self):
|
|
"""
|
|
"""
|
|
# FIXME: Put really general stuff here !
|
|
self.videospeedlistChoices = [u"slow", u"medium", u"fast"]
|
|
self.gaugerange = 100
|
|
|
|
self.root_fo = ""
|
|
self.mode = "crop" # type of video to be created
|
|
self.sort = "name" # how image files will be chronologically sorted
|
|
self.speed = 1 # Speed of the movie
|
|
self.param = "frontal_face" # type of face profile to be searched for
|
|
|
|
if "win" in sys.platform:
|
|
self.out_fo = "C:/Users/jll/Videos/" # Default folder for Windows
|
|
self.in_fo = "C:\Users\jll\Pictures/"
|
|
else:
|
|
self.out_fo = "/home/jll/Videos/" # Default folder for Linux
|
|
self.in_fo = "/home/jll/Pictures/"
|
|
|
|
# Overriding event handling methods
|
|
def on_settings(self, event):
|
|
settings = SettingsWindow(self)
|
|
settings.Show(True) # Finally show the frame
|
|
|
|
def on_start(self, event):
|
|
"""
|
|
User asks for starting the algorithm
|
|
Sets all parameters and start processing
|
|
"""
|
|
self.my_logger.debug("start pressed")
|
|
|
|
if not self.process_running: # start only if not already running
|
|
# Empty list on screen
|
|
self.filelist.DeleteAllItems()
|
|
|
|
self.set_parameters()
|
|
self.print_parameters()
|
|
# Instantiating the facemovie
|
|
self.facemovie = FacemovieThread.FacemovieThread(self.face_params)
|
|
self.facemovie.subscribe(self) # I want new information ! Subscribes to facemovie reports
|
|
self.subscribe(self.facemovie) # Subscribing facemovie to our messages
|
|
|
|
self.facemovie.start()
|
|
|
|
self.process_running = True
|
|
else:
|
|
self.console_logger.error("Cannot start, process already running !")
|
|
self.my_logger.error("Cannot start, process already running !")
|
|
|
|
def on_stop(self, event):
|
|
"""
|
|
User asks for stopping the algorithm
|
|
Asks the FacemovieThread to terminate
|
|
"""
|
|
self.my_logger.debug("Stop pressed")
|
|
self.console_logger.debug("Stop pressed")
|
|
self.notify(["Application", ["STOP"]]) # Asking the Facemovie to stop
|
|
self.process_running = False
|
|
|
|
#self.on_exit(event) # Finally shuts down the interface
|
|
|
|
def on_input(self, event):
|
|
"""
|
|
Activated when a user clicks to choose its input location
|
|
"""
|
|
self.inputdialog = wx.DirDialog(self, "Please choose your input directory", style=1, defaultPath=self.in_fo)
|
|
|
|
if self.inputdialog.ShowModal() == wx.ID_OK:
|
|
self.in_fo = self.inputdialog.GetPath()
|
|
self.inputtextbox.SetLabel(self.in_fo)
|
|
self.inputdialog.Destroy()
|
|
|
|
def on_output(self, event):
|
|
"""
|
|
Activated when a user clicks to choose its output location
|
|
"""
|
|
default_dir = "~/Videos"
|
|
self.outputdialog = wx.DirDialog(self, "Please choose your output directory", style=1, defaultPath=default_dir)
|
|
|
|
if self.outputdialog.ShowModal() == wx.ID_OK:
|
|
self.outputchoosertext.SetLabel(self.outputdialog.GetPath())
|
|
self.outputdialog.Destroy()
|
|
|
|
def on_help(self, event):
|
|
"""
|
|
Opens a browser and points to online help.
|
|
"""
|
|
url = "http://jlengrand.github.com/FaceMovie/"
|
|
webbrowser.open(url, new=2) # in new tab if possible
|
|
|
|
def on_about(self, event):
|
|
"""
|
|
Displays the about box for Ivolution
|
|
"""
|
|
description =""" Ivolution is a project aiming at helping you
|
|
create videos of yourself over time.
|
|
Simply take pictures of yourself, Ivolution does
|
|
everything else for you.
|
|
|
|
Ivolution may be used for faces, but also profiles
|
|
(to show women along pregnancy for example)
|
|
or full body (for people workouting).
|
|
|
|
The only limitation comes from you !
|
|
"""
|
|
|
|
licence = """Copyright (c) <2012>, <Julien Lengrand-Lambert>
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright notice, this
|
|
list of conditions and the following disclaimer.
|
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
|
this list of conditions and the following disclaimer in the documentation
|
|
and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
The views and conclusions contained in the software and documentation are those
|
|
of the authors and should not be interpreted as representing official policies,
|
|
either expressed or implied, of the FreeBSD Project."""
|
|
|
|
info = wx.AboutDialogInfo()
|
|
|
|
info.SetIcon(wx.Icon('ivolution/data/media/vitruve.png', wx.BITMAP_TYPE_PNG))
|
|
info.SetName('Ivolution')
|
|
info.SetVersion('0.8')
|
|
info.SetDescription(description)
|
|
info.SetCopyright('(C) 2012 Julien Lengrand-Lambert')
|
|
info.SetWebSite('http://www.lengrand.fr')
|
|
info.SetLicence(licence)
|
|
info.AddDeveloper('Julien Lengrand-Lambert')
|
|
info.AddDocWriter('Julien Lengrand-Lambert')
|
|
info.AddArtist('Luc Viatour')
|
|
info.AddTranslator('Julien Lengrand-Lambert')
|
|
|
|
wx.AboutBox(info)
|
|
|
|
def on_exit(self, event):
|
|
"""
|
|
Called when the IvolutionWindow is closed, or File/Exit is called.
|
|
"""
|
|
# Clean up code for saving application state should be added here.
|
|
self.notify(["Application", ["STOP"]]) # Asking the Facemovie to stop
|
|
self.process_running = False
|
|
self.Close(True) # Close the frame.
|
|
|
|
# system methods
|
|
def get_current_mode(self):
|
|
"""
|
|
"""
|
|
if self.cropmode.GetValue():
|
|
mode = "crop"
|
|
else:
|
|
mode = "conservative"
|
|
|
|
return mode
|
|
|
|
def get_current_sort(self):
|
|
"""
|
|
"""
|
|
if self.namemode.GetValue():
|
|
sort = "name"
|
|
else:
|
|
sort = "exif"
|
|
|
|
return sort
|
|
|
|
def set_parameters(self):
|
|
"""
|
|
Retrieves all parameters needed for the algorithm to run
|
|
"""
|
|
#self.in_fo = self.inputtextbox.GetLabel() + "/"
|
|
# self.in_fo = "C:\Users\jll\perso\Ivolution\ivolution\data\samples" + "/"
|
|
# self.out_fo = "C:\Users\jll\Videos" + "/"
|
|
# #self.out_fo = self.outputchoosertext.GetLabel() + "/"
|
|
# self.param = "frontal_face"
|
|
# #self.param = self.typefacelist.GetValue()
|
|
# #self.speed = self.videospeedlistChoices.index(self.videospeedlist.GetValue()) # We need and integer between 0 and 2
|
|
# self.speed = self.videospeedlistChoices[1]
|
|
|
|
# self.mode = "crop"
|
|
# self.sort = "name"
|
|
#self.mode = self.get_current_mode()
|
|
#self.sort = self.get_current_sort()
|
|
self.print_parameters()
|
|
|
|
# Instantiating the face_params object that will be needed by the facemovie
|
|
par_fo = os.path.join(self.root_fo, get_data("haarcascades"))
|
|
self.face_params = FaceParams.FaceParams(par_fo,
|
|
self.in_fo,
|
|
self.out_fo,
|
|
self.param,
|
|
self.sort,
|
|
self.mode,
|
|
self.speed)
|
|
|
|
def print_parameters(self):
|
|
print "#########"
|
|
print "Settings:"
|
|
print "input folder : %s" % (self.in_fo)
|
|
print "output folder : %s" % (self.out_fo)
|
|
|
|
print "Face Type : %s" % (self.param)
|
|
print "Speed chosen : %s" % (self.speed)
|
|
print "Mode chosen : %s" % (self.mode)
|
|
print "Sort method : %s" % (self.sort)
|
|
|
|
print "#########"
|
|
|
|
def setup_logger(self):
|
|
"""
|
|
Configures our logger to save error messages
|
|
Start logging in file here
|
|
"""
|
|
personal_dir = "~/.ivolution"
|
|
log_root = 'fm.log'
|
|
log_file = os.path.join(os.path.expanduser(personal_dir), log_root)
|
|
|
|
# create logger for 'facemovie'
|
|
self.my_logger = logging.getLogger('FileLog')
|
|
|
|
self.my_logger.setLevel(logging.DEBUG)
|
|
# create file handler which logs even debug messages
|
|
|
|
#fh = logging.StreamHandler()
|
|
fh = logging.FileHandler(log_file)
|
|
|
|
fh.setLevel(logging.DEBUG)
|
|
# create console handler with a higher log level
|
|
self.console_logger = logging.getLogger('ConsoleLog')
|
|
self.console_logger.setLevel(logging.DEBUG) # not needed
|
|
|
|
ch = logging.StreamHandler()
|
|
#ch.setLevel(logging.DEBUG) # not needed
|
|
|
|
# add the handlers to the logger
|
|
self.my_logger.addHandler(fh)
|
|
|
|
self.my_logger.info("######") # Separating different sessions
|
|
|
|
formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')
|
|
# create formatter and add it to the handlers
|
|
fh.setFormatter(formatter)
|
|
#ch.setFormatter(formatter)
|
|
|
|
self.console_logger.addHandler(ch)
|
|
|
|
def update(self, message):
|
|
"""
|
|
Trigerred by FacemovieThread.
|
|
Uses the Observer pattern to inform the user about the progress of the current job.
|
|
"""
|
|
if len(message) == 3:
|
|
# notifications
|
|
#self.console_logger.debug(message)
|
|
self.my_logger.debug(message)
|
|
|
|
if message[0] == "PROGRESS": # progress bar
|
|
# big steps performed
|
|
wx.MutexGuiEnter() # to avoid thread problems
|
|
self.progressgauge.SetValue(self.gaugerange * float(message[2]))
|
|
self.statusbar.SetStatusText(message[1], 0)
|
|
wx.MutexGuiLeave()
|
|
|
|
if float(message[2]) >= 1.0: # 100% of process
|
|
self.my_logger.debug("Reached end of facemovie process")
|
|
#self.console_logger.debug("Reached end of facemovie process")
|
|
self.process_running = False
|
|
|
|
elif message[0] == "STATUS": # status label
|
|
if message[1] == "Error":
|
|
wx.MutexGuiEnter() # to avoid thread problems
|
|
self.statusbar.SetStatusText("Error detected", 0)
|
|
self.progressgauge.SetValue(0)
|
|
wx.MutexGuiLeave()
|
|
self.process_running = False
|
|
|
|
wx.MutexGuiEnter() # to avoid thread problems
|
|
self.statusbar.SetStatusText(message[1], 1)
|
|
wx.MutexGuiLeave()
|
|
elif message[0] == "FILEADD":
|
|
item = wx.ListItem()
|
|
item.SetText(message[1])
|
|
wx.MutexGuiEnter() # to avoid thread problems
|
|
self.filelist.InsertItem(item)
|
|
wx.MutexGuiLeave()
|
|
elif message[0] == "FILEDONE":
|
|
for i in range(self.filelist.GetItemCount()):
|
|
if message[1] == self.filelist.GetItemText(i):
|
|
if message[2] == 1:
|
|
color = "green"
|
|
else:
|
|
color = "red"
|
|
wx.MutexGuiEnter() # to avoid thread problems
|
|
self.filelist.SetItemTextColour(i, color)
|
|
wx.MutexGuiLeave()
|
|
|
|
elif len(message) > 1: # system commands shall be ignored
|
|
self.console_logger.debug("Unrecognized command")
|
|
self.my_logger.debug("Unrecognized command")
|
|
self.console_logger.debug(message)
|
|
self.my_logger.debug(message)
|