2016-07-16 7 views
8

Я хочу сгенерировать некоторые gif с прозрачным фоном, используя matplotlib. Я пробовал разные варианты, но я не могу получить файлы с прозрачным фоном. С текущей настройкой я получаю первый кадр, но не все остальные. Ниже мой кодmplot3d анимация с прозрачным фоном

from __future__ import division 
from numpy import pi, sin, cos, mgrid 
from scipy.special import jn, jn_zeros 
from mpl_toolkits.mplot3d import Axes3D 
import matplotlib.pyplot as plt 
import matplotlib.animation as animation 
from matplotlib import rcParams 


# In Windows the next line should provide the full path to convert.exe 
# since convert is a Windows command 
rcParams['animation.convert_path'] = "C:\Program Files\ImageMagick-6.9.3\convert.exe" 
rcParams['savefig.transparent'] = True 
rcParams['savefig.dpi'] = 130 
rcParams['savefig.pad_inches'] = 0 
plot_args = {'rstride': 1, 'cstride': 1, 'cmap':"RdYlBu", 
      'linewidth': 0.5, 'antialiased': True, 'color': '#1e1e1e', 
      'shade': True, 'alpha': 1.0, 'vmin': -1, 'vmax':1} 


def data_gen(num): 
    ax.cla() 
    m, n = 1, 2 
    lam = jn_zeros(m, n)[-1] 
    dt = 2*pi/(30*lam) 
    z = cos(m*t)*jn(m, lam*r)*sin(lam*num*dt) 
    surf = ax.plot_surface(x, y, z, **plot_args) 
    ax.view_init(elev=30, azim=45) 
    ax.set_xlim(-0.6, 0.6) 
    ax.set_ylim(-0.6, 0.6) 
    ax.set_zlim(-1, 1) 
    plt.axis("off") 
    return surf 


r, t = mgrid[0:1:20j, 0:2*pi:40j] 
x, y = r*cos(t), r*sin(t) 
fig = plt.figure(facecolor=None) 
ax = fig.add_subplot(111, projection='3d') 
ani = animation.FuncAnimation(fig, data_gen, range(30), blit=False) 
ani.save("Drum vibration mode.gif", writer='imagemagick') 

Это дает в результате (если нажать на нее, вы можете увидеть, что только один кадр является прозрачным)

Вопрос: Есть ли способ, чтобы получить анимацию с прозрачным фоном, используя mplot3d?

enter image description here

+0

Вы видели это http://blog.adrian.pw/posts/2016/06/matplotlib-transparent-animation/? Кажется, что вы ищете 'fig.patch.set_alpha (0.)' и что-то вроде 'ani.save ('circle_anim.mov', codec =" png ", dpi = 100, bitrate = -1, savefig_kwargs = {'transparent': True, 'facecolor': 'none'}) '. Я не могу попробовать прямо сейчас, но позже пойду. –

+0

@JacquesGaudin, я только что проверил. Это классно, но он не экспортирует изображение как GIF. Действительно, '' savefig_kwargs'' игнорируются при использовании GIF-записи. – nicoguaro

+0

@JacquesGaudin, просто попробовал: не работает. – nicoguaro

ответ

4

Я думаю, что это действительно ошибка. Однако, если вы скорее заботитесь о результате, чем о том, как туда добраться, следующее будет выполнять эту работу. Вместо вызова анимации вы можете сохранить каждое изображение отдельно, а затем вызвать imageMagick, чтобы преобразовать их в анимированный gif. См. Код ниже и рассмотрите аргументы convert.exe.

from __future__ import division 
from numpy import pi, sin, cos, mgrid 
from scipy.special import jn, jn_zeros 
from mpl_toolkits.mplot3d import Axes3D 
import matplotlib.pyplot as plt 

from matplotlib import rcParams 
import subprocess 


# In Windows the next line should provide the full path to convert.exe 
# since convert is a Windows command 
path_to_convert = "C:\Program Files\ImageMagick-6.9.3\convert.exe" 
#rcParams['animation.convert_path'] = path_to_convert 
rcParams['savefig.transparent'] = True 
rcParams['savefig.dpi'] = 130 
rcParams['savefig.pad_inches'] = 0 
plot_args = {'rstride': 1, 'cstride': 1, 'cmap':"RdYlBu", 
      'linewidth': 0.5, 'antialiased': True, 'color': '#1e1e1e', 
      'shade': True, 'alpha': 1.0, 'vmin': -1, 'vmax':1} 


def data_gen(num): 
    ax.cla() 
    ax.clear() 
    m, n = 1, 2 
    lam = jn_zeros(m, n)[-1] 
    dt = 2*pi/(30*lam) 
    z = cos(m*t)*jn(m, lam*r)*sin(lam*num*dt) 
    surf = ax.plot_surface(x, y, z, **plot_args) 
    ax.view_init(elev=30, azim=45) 
    ax.set_xlim(-0.6, 0.6) 
    ax.set_ylim(-0.6, 0.6) 
    ax.set_zlim(-1, 1) 
    ax.axis("off") 
    fig.patch.set_visible(False) 
    ax.patch.set_visible(False) 
    ax.set_axis_off() 
    ax._axis3don = False 
    return surf 


r, t = mgrid[0:1:20j, 0:2*pi:40j] 
x, y = r*cos(t), r*sin(t) 
fig = plt.figure(facecolor=None, frameon=False) 
ax = fig.add_subplot(111, projection='3d') 
for i in range(30): 
    data_gen(i) 
    plt.savefig("drum_{n:02d}.png".format(n=i), transparent=True, frameon=False) 
    print i, 

args = [path_to_convert, "-delay", "10", "-loop" , "0", "-dispose", "Background", "drum_*.png", "output.gif"] 
subprocess.call(args, shell=True) 
subprocess.call(["del", "/Q", "drum_*.png"], shell=True) 
print "\ndone" 

Помните, что вызовы imageMagick и команда delete могут быть зависимыми от системы. Этот скрипт был протестирован с помощью imageMagick 7.0.3 (Q16) под Windows 8.

+0

Я думал об этом варианте, но был надеясь, что кто-то сможет сделать это по-другому. Чтобы ответить на ваш вопрос, я изменил строку '' args'' на '' args = [path_to_convert, "drum _ * .png", "-delay", "10", "-loop", "0", " dispose "," Background "," output.gif "]' '. – nicoguaro

+0

Было бы здорово, если бы мы могли добавить что-то вроде '' subprocess.call (["rm", "drum _ * .png"], shell = False) '', чтобы стереть PNG-файлы и просто сохранить GIF. – nicoguaro

+0

Я добавил строку, чтобы избавиться от файлов png. – ImportanceOfBeingErnest

Смежные вопросы