简体   繁体   English

动画散点图和等高线图

[英]Animate scatter and contour plot

I am trying to generate an animated scatter plot within a functioning animated contour plot .我想产生一个animated scatter plot一个运作中animated contour plot I can get both working separately but not together.我可以让两者分开工作,但不能一起工作。

The code below generates the contour from coordinates A and B in the df .下面的代码从df中的坐标AB生成contour I've attempted to include a separated animated scatter within the sample plot using the C coordinates.我尝试使用C坐标在示例plot包含一个单独的动画scatter This attempt is currently commented out.此尝试目前已被注释掉。

So I basically want to include another animated scatter using C_X and C_Y .所以我基本上想使用C_XC_Y包含另一个animated scatter I've trying by applying them to line_c .我尝试将它们应用于line_c

import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
import scipy.stats as sts
import matplotlib.animation as animation
import matplotlib.transforms as transforms

''' Section 1 '''

DATA_LIMITS = [-85, 85]

def datalimits(*data):
    return DATA_LIMITS  # dmin - spad, dmax + spad

def mvpdf(x, y, xlim, ylim, radius=1, velocity=0, scale=0, theta=0):
    X,Y = np.meshgrid(np.linspace(*xlim), np.linspace(*ylim))
    XY = np.stack([X, Y], 2)
    PDF = sts.multivariate_normal([x, y]).pdf(XY)
    return X, Y, PDF

def mvpdfs(xs, ys, xlim, ylim, radius=None, velocity=None, scale=None, theta=None):
    PDFs = []
    for i,(x,y) in enumerate(zip(xs,ys)):
        X, Y, PDF = mvpdf(x, y, xlim, ylim)
        PDFs.append(PDF)

    return X, Y, np.sum(PDFs, axis=0)

''' Animate Plot '''

fig, ax = plt.subplots(figsize = (10,6))
ax.set_xlim(DATA_LIMITS)
ax.set_ylim(DATA_LIMITS)

#Animated coordinates for group A,B
line_a, = ax.plot([], [], '.', c='red', alpha = 0.5, markersize=5, animated=True)
line_b, = ax.plot([], [], '.', c='blue', alpha = 0.5, markersize=5, animated=True)

#Attempt to incorporate scatter for C 
line_c, = ax.plot([], [], '.', c='white', alpha = 0.5, markersize=2.5, animated=True)

cfs = None

def plotmvs(tdf, xlim=None, ylim=None, fig=fig, ax=ax):
    global cfs  
    if cfs:
        for tp in cfs.collections:
            # Remove the existing contours
            tp.remove()

    # Get the data frame for time t
    df = tdf[1]

    if xlim is None: xlim = datalimits(df['X'])
    if ylim is None: ylim = datalimits(df['Y'])

    PDFs = []

    for (group, gdf), group_line in zip(df.groupby('group'), (line_a, line_b)):
        group_line.set_data(*gdf[['X','Y']].values.T)
        X, Y, PDF = mvpdfs(gdf['X'].values, gdf['Y'].values, xlim, ylim)
        PDFs.append(PDF)

    normPDF = PDF - PDF.min()
    normPDF = normPDF / normPDF.max()
    cfs = ax.contourf(X, Y, normPDF, cmap='viridis', alpha = 1, levels=np.linspace(-1,1,10))

#Create offset scatter for Group C
#    for (group, g2df), group_line in zip(df.groupby('group'), (line_c)):    
#        group_line.set_data(*g2df[['XX','YY']].values.T)

#        offset = lambda p: transforms.ScaledTranslation(p/82.,0, plt.gcf().dpi_scale_trans)
#        trans = plt.gca().transData
#        ax.scatter(line_c,transform=trans+offset(+2))


    return cfs.collections + [line_a, line_b]#, line_c] 

n = 10
time = range(n)  

d = ({
     'A1_X' :    [13.3,13.16,12.99,12.9,12.79,12.56,12.32,12.15,11.93,11.72],
     'A1_Y' :    [26.12,26.44,26.81,27.18,27.48,27.82,28.13,28.37,28.63,28.93],
     'A2_X' :    [6.97,6.96,7.03,6.98,6.86,6.76,6.55,6.26,6.09,5.9],
     'A2_Y' :    [10.92,10.83,10.71,10.52,10.22,10.02,9.86,9.7,9.54,9.37],
     'B1_X' :    [38.35,38.1,37.78,37.55,37.36,37.02,36.78,36.46,36.21,35.79],
     'B1_Y' :    [12.55,12.58,12.58,12.55,12.5,12.47,12.43,12.48,12.44,12.44],
     'B2_X' :    [14.6,14.38,14.16,13.8,13.45,13.11,12.71,12.3,12.06,11.61],
     'B2_Y' :    [4.66,4.44,4.24,4.1,4.01,3.84,3.67,3.56,3.44,3.47],
#    'C_X' :    [10,15,18,20,30,33,35,42,34,20],
#    'C_Y' :    [10,16,20,10,20,13,15,12,14,10],                 
     })

tuples = [((t, k.split('_')[0][0], int(k.split('_')[0][1:]), k.split('_')[1]), v[i]) 
      for k,v in d.items() for i,t in enumerate(time)]

df = pd.Series(dict(tuples)).unstack(-1)
df.index.names = ['time', 'group', 'id']

interval_ms = 200
delay_ms = 1000
ani = animation.FuncAnimation(fig, plotmvs, frames=df.groupby('time'),
            blit=True, interval=interval_ms, repeat_delay=delay_ms)

plt.show()

Ok, so I had to change a few things:好的,所以我不得不改变一些事情:

  • Most importanty, I turned off blit=True (mostly because I'm on a Mac, where this isn't fully supported, so your mileage may vary) and animated=True in the definitions of line_a and line_b .最重要的是,我在line_aline_b的定义中关闭了blit=True (主要是因为我在 Mac 上,不完全支持它,所以你的里程可能会有所不同)和line_a animated=True
  • The group "C" that you had created in your code comments was missing an integer for the tuples=[…] code to work (it expects to find an int(k.split('_')[0][1:] )您在代码注释中创建的组“C”缺少一个整数,以便tuples=[…]代码工作(它期望找到一个int(k.split('_')[0][1:] )
  • The scatter plot should be a .scatter() , and since you're not using an init_func=init call within the FuncAnimation , we can simply create it "on the fly", within the animation function itself.散点图应该是一个.scatter()因为你使用的不是init_func=init的内通话FuncAnimation ,我们可以简单地创建它“对飞”,动画功能本身。
  • I've added a second for loop with an if group=='C' case — though you could solve this more elegantly — that creates the scatterplot scat .我在if group=='C'情况下添加了第二个for循环——尽管你可以更优雅地解决这个问题——它创建了散点图scat Note that I included the transform, although I'm not sure what this will achieve for you, in the end请注意,我包含了转换,尽管我不确定这最终会为您带来什么
  • I set the levels in ax.contourf() to the interval [0,1], but that's completely pedantic from my side ;-)我将ax.contourf()的级别设置为区间 [0,1],但从我这边来看,这完全是迂腐的 ;-)
  • I've added zorder= in all plot items in order to control in which z-plane they are plotted (optional, but helpful), and slightly adjusted the look of the plotted lines (optional, only for emphasis)我在所有绘图项中添加了zorder=以控制它们绘制在哪个 z 平面(可选,但有用),并稍微调整了绘制线的外观(可选,仅用于强调)
  • In the end, you need to make sure you're returning an iterable list , hence the perhaps "odd looking" return cfs.collections + [scat] + [line_a,line_b]最后,您需要确保返回一个可迭代列表,因此可能“看起来很奇怪” return cfs.collections + [scat] + [line_a,line_b]

Plot result:绘图结果:

完整的动画

Complete code:完整代码:

import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
import scipy.stats as sts
import matplotlib.animation as animation
import matplotlib.transforms as transforms
from matplotlib.lines import Line2D

''' Section 1 '''

DATA_LIMITS = [-85, 85]

def datalimits(*data):
    return DATA_LIMITS  # dmin - spad, dmax + spad

def mvpdf(x, y, xlim, ylim, radius=1, velocity=0, scale=0, theta=0):
    X,Y = np.meshgrid(np.linspace(*xlim), np.linspace(*ylim))
    XY = np.stack([X, Y], 2)
    PDF = sts.multivariate_normal([x, y]).pdf(XY)
    return X, Y, PDF

def mvpdfs(xs, ys, xlim, ylim, radius=None, velocity=None, scale=None, theta=None):
    PDFs = []
    for i,(x,y) in enumerate(zip(xs,ys)):
        X, Y, PDF = mvpdf(x, y, xlim, ylim)
        PDFs.append(PDF)

    return X, Y, np.sum(PDFs, axis=0)

''' Animate Plot '''

fig, ax = plt.subplots(figsize = (10,6))
ax.set_xlim(DATA_LIMITS)
ax.set_ylim(DATA_LIMITS)

#Animated coordinates for group A,B
line_a, = ax.plot([], [], '-o', c='red', alpha = 0.5, markersize=5,zorder=3)
line_b, = ax.plot([], [], '-o', c='blue', alpha = 0.5, markersize=5,zorder=3)

cfs = None

def plotmvs(tdf, xlim=None, ylim=None, fig=fig, ax=ax):    
    global cfs  
    if cfs:
        for tp in cfs.collections:
            # Remove the existing contours
            tp.remove()

    # Get the data frame for time t
    df = tdf[1]

    if xlim is None: xlim = datalimits(df['X'])
    if ylim is None: ylim = datalimits(df['Y'])

    PDFs = []

    for (group, gdf), group_line in zip(df.groupby('group'), (line_a, line_b)):
        group_line.set_data(*gdf[['X','Y']].values.T)
        X, Y, PDF = mvpdfs(gdf['X'].values, gdf['Y'].values, xlim, ylim)
        PDFs.append(PDF)

    for (group, gdf) in df.groupby('group'):
        if group=='C':
            offset = lambda p: transforms.ScaledTranslation(p/82.,0, plt.gcf().dpi_scale_trans)
            trans = plt.gca().transData
            scat=ax.scatter(gdf['X'].values, gdf['Y'].values, 
                    marker='o', c='white', alpha = 0.5,zorder=3,
                    transform=trans+offset(+2) )#markersize=2,zorder=3)


    normPDF = PDF - PDF.min()
    normPDF = normPDF / normPDF.max()

    cfs = ax.contourf(X, Y, normPDF, cmap='viridis', alpha = 1, levels=np.linspace(0,1,10),zorder=1)

    return  cfs.collections + [scat] + [line_a,line_b] # make sure that these are iterable!

n = 10
time = range(n)  

d = ({
    'A1_X' :    [13.3,13.16,12.99,12.9,12.79,12.56,12.32,12.15,11.93,11.72],
    'A1_Y' :    [26.12,26.44,26.81,27.18,27.48,27.82,28.13,28.37,28.63,28.93],
    'A2_X' :    [6.97,6.96,7.03,6.98,6.86,6.76,6.55,6.26,6.09,5.9],
    'A2_Y' :    [10.92,10.83,10.71,10.52,10.22,10.02,9.86,9.7,9.54,9.37],
    'B1_X' :    [38.35,38.1,37.78,37.55,37.36,37.02,36.78,36.46,36.21,35.79],
    'B1_Y' :    [12.55,12.58,12.58,12.55,12.5,12.47,12.43,12.48,12.44,12.44],
    'B2_X' :    [14.6,14.38,14.16,13.8,13.45,13.11,12.71,12.3,12.06,11.61],
    'B2_Y' :    [4.66,4.44,4.24,4.1,4.01,3.84,3.67,3.56,3.44,3.47],
    'C1_X' :    [10.,15.,18.,20.,30.,33.,35.,42.,34.,20.],## name contains an int so that tuples=... list works!
    'C1_Y' :    [10.,16.,20.,10.,20.,13.,15.,12.,14.,10.],
})

tuples = [((t, k.split('_')[0][0], int(k.split('_')[0][1:]), k.split('_')[1]), v[i])
    for k,v in d.items() for i,t in enumerate(time) ]

df = pd.Series(dict(tuples)).unstack(-1)
df.index.names = ['time', 'group', 'id']

interval_ms = 200
delay_ms = 1000
ani = animation.FuncAnimation(fig, plotmvs,  frames=df.groupby('time'), interval=interval_ms, repeat_delay=delay_ms,)

plt.show()

Update更新

If you'd rather have a single moving point , at the same time minimising/streamlining the lines of code, you could also initiate the plot elements with:如果您希望有一个单一的移动点,同时最小化/简化代码行,您还可以使用以下命令启动绘图元素:

line_a, = ax.plot([], [], '-o', c='red', alpha = 0.5, markersize=5,zorder=3)
line_b, = ax.plot([], [], '-o', c='blue', alpha = 0.5, markersize=5,zorder=3)
lines=[line_a,line_b] ## this is iterable!

offset = lambda p: transforms.ScaledTranslation(p/82.,0, plt.gcf().dpi_scale_trans)
trans = plt.gca().transData

scat = ax.scatter([], [], s=5**2,marker='o', c='white', alpha = 0.5,zorder=3,transform=trans+offset(+2) )
scats=[scat] ## this is iterable, too!

and then inside plotmvs():然后在 plotmvs() 中:

for (group, gdf), group_line in zip(df.groupby('group'), lines+scats):
        if group in ['A','B']:
            group_line.set_data(*gdf[['X','Y']].values.T)
            X, Y, PDF = mvpdfs(gdf['X'].values, gdf['Y'].values, xlim, ylim)
            PDFs.append(PDF)
        elif group in ['C']:
            x,y=(gdf['X'].values, gdf['Y'].values)
            scat.set_offsets( gdf[['X','Y']].values )

Note that updating scatter plots and line plots uses different functions and coordinate lists!请注意,更新散点图和折线图使用不同的函数和坐标列表!

And finally update the return values:最后更新返回值:

return  cfs.collections + scats + lines # make sure that these are iterable!

Which yields the following animation:产生以下动画:

次要绘图选项

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM