/face-movie

Automatically create a time-lapse morph sequence of a face using OpenCV and Dlib.

Primary LanguagePython

face-movie

Create a video warp sequence of human faces. Can be used, for example, to create a time-lapse video showing someone's face change over time. See demos here and here.

Supported on Python 3 and OpenCV 3+. Tested on macOS High Sierra and Mojave.

Requirements

  • OpenCV
    • For conda users, run conda install -c conda-forge opencv.
  • Dlib
    • For conda users, run conda install -c conda-forge dlib.
  • ffmpeg
    • For conda users, run conda install -c conda-forge ffmpeg.
  • scipy
  • numpy
  • matplotlib
  • pillow

Installation

  1. Clone the repo.
git clone https://github.com/andrewdcampbell/face-movie
  1. Download the trained face detector model from here. Unzip it and place it in the root directory of the repo.

Creating a face movie - reccomended workflow

  1. Make a directory <FACE_MOVIE_DIR> in the root directory of the repo with the desired face images. The images must feature a clear frontal view of the desired face (other faces can be present too). The image filenames must be in lexicographic order of the order in which they are to appear in the video.

  2. Create a directory <ALIGN_OUTPUT>. Then align the faces in the images with

python face-movie/align.py -images <FACE_MOVIE_DIR> -target <BASE_IMAGE> 
                           [-overlay] [-border <BORDER>] -outdir <ALIGN_OUTPUT>

The output will be saved to the provided <ALIGN_OUTPUT> directory. BASE_IMAGE is the image to which all other images will be aligned to. It should represent the "typical" image of all your images - it will determine the output dimensions and facial position.

The optional -overlay flag places subsequent images on top of each other (recommended). The optional -border <BORDER> argument adds a white border <BORDER> pixels across to all the images for aesthetics. I think around 5 pixels looks good.

If your images contain multiple faces, a window will appear with the faces annotated and you will be prompted to enter the index of the correct face on the command line.

At this point you should inspect the output images and re-run the alignment with new parameters until you're satisfied with the result.

  1. Morph the sequence with
python face-movie/main.py -morph -images <ALIGN_OUTPUT> -td <TRANSITION_DUR> 
                          -pd <PAUSE_DUR> -fps <FPS> -out <OUTPUT_NAME>.mp4

This will create a video OUTPUT_NAME.mp4 in the root directory with the desired parameters. Note that TRANSITION_DUR and PAUSE_DUR are floating point values while FPS is an integer.

You may again be prompted to choose the correct face.

  1. (Optional) Add music with
ffmpeg -i <OUTPUT_NAME>.mp4 -i <AUDIO_TRACK> -map 0:v -map 1:a -c copy 
       -shortest -pix_fmt yuv420p <NEW_OUTPUT_NAME>.mov

The images used to produce the first demo video are included in demos/daniel. Step 2 should produce something like the aligned images shown in demos/daniel_aligned. Step 3 should produce something like the video daniel.mp4.

Notes

  • If you have photos that are already the same size and all of the faces are more or less aligned already (e.g. roster photos or yearbook photos), you can skip steps 1 and 2 and just use the directory of photos as your aligned images.

  • If you want to add some photos to the end of an existing morph sequence (and avoid rendering the entire video again), you can make a temporary directory containing the last aligned image in the movie and aligned images you want to append. Make a morph sequence out of just those images, making sure to use the same frame rate. Now concatenate them with

ffmpeg -safe 0 -f concat -i list.txt -c copy merged.mp4

where list.txt is of the form

file 'path/to/video1.mp4'
file 'path/to/video2.mp4'
...

  • For adding music longer than the video, you can fade the audio out with
VIDEO="<PATH_TO_VIDEO>"
AUDIO="<PATH_TO_AUDIO>" # can also drag and drop audio track to terminal (don't use quotes)
FADE="<FADE_DURATION_IN_SECONDS>"

VIDEO_LENGTH="$(ffprobe -v error -select_streams v:0 -show_entries stream=duration -of csv=p=0 $VIDEO)"
ffmpeg -i $VIDEO -i "$AUDIO" -filter_complex "[1:a]afade=t=out:st=$(bc <<< "$VIDEO_LENGTH-$FADE"):d=$FADE[a]" -map 0:v:0 -map "[a]" -c:v copy -c:a aac -shortest with_audio.mp4

Averaging Faces

You can also use the code to create a face average. Follow the same steps 1) - 2) as above. You probably don't want to overlay images or use a border, however. Then run

python face-movie/main.py -average -images <ALIGN_OUTPUT> -out <OUTPUT_NAME>.jpg

A small face dataset is included in the demos directory.

The computed average male and female face are shown below.

Acknowledgements