A simple PyTorch Implementation of Generative Adversarial Networks, focusing on anime face drawing.
The images are generated from a DCGAN model trained on 143,000 anime character faces for 100 epochs.
Manipulating latent codes, enables the transition from images in the first row to the last row.
The images are not clean, some outliers can be observed, which degrades the quality of the generated images.
To run the experiment,
$ python main.py --dataRoot path_to_dataset/
The pretrained model for DCGAN are also in this repo, play it inside the jupyter notebook.
Anime-style images of 126 tags are collected from danbooru.donmai.us using the crawler tool gallery-dl. The images are then processed by a anime face detector python-animeface. The resulting dataset contains ~143,000 anime faces. Note that some of the tags may no longer meaningful after cropping, i.e. the cropped face images under 'uniform' tag may not contain visible parts of uniforms.
How to construct the dataset from scratch ?
Prequisites: gallery-dl, python-animeface
-
Download anime-style images
# download 1000 images under the tag "misaka_mikoto" gallery-dl --images 1000 "https://danbooru.donmai.us/posts?tags=misaka_mikoto" # in a multi-processing manner cat tags.txt | \ xargs -n 1 -P 12 -I 'tag' \ bash -c ' gallery-dl --images 1000 "https://danbooru.donmai.us/posts?tags=$tag" '
-
Extract faces from the downloaded images
import animeface from PIL import Image im = Image.open('images/anime_image_misaka_mikoto.png') faces = animeface.detect(im) x,y,w,h = faces[0].face.pos im = im.crop((x,y,x+w,y+h)) im.show() # display
I've cleaned the original dataset, the new version of the dataset has 115085 images in 126 tags. You can access the images from:
- GitHub: https://github.com/jayleicn/animeGAN/releases/tag/data
- BaiduYun: https://pan.baidu.com/s/1o8Nxllo
Non-commercial use please.
- GANs are really hard to train.
- DCGAN generally works well, simply add fully-connected layers causes problems.
- In my cases, more layers for G yields better images, in the sense that G should be more powerful than D.
- Add noise to D's inputs and labels helps stablize training.
- Use differnet input and generate resolution (64x64 vs 96x96), there seems no obvious difference during training, the generated images are also very similar.
- Binray Noise as G's input amazingly works, but the images are not as good as those with Gussian Noise, idea credit to @cwhy ['Binary Noise' here I mean a sequence of {-1,1} generated by bernoulli distribution at p=0.5 ]
I did not carefully verify them, if you are looking for some general GAN tips, see @soumith's ganhacks
- This project is heavily influenced by chainer-DCGAN and IllustrationGAN, the codes are mostly borrowed from PyTorch DCGAN example, thanks the authors for the clean codes.
- Dependencies: pytorch, torchvision
- This is a toy project for me to learn PyTorch and GANs, most importantly, for fun! :) Any feedback is welcome.
@jayleicn