python-在matplotlib中用中点箭头绘制圆形的fancyarrowpatch

我一直在尝试扩大matplotlib补丁的边界,并指示它绘制一个带有中点方向箭头的圆形FancyArrowPatch.在我尝试创建的网络表示中,这将被证明非常有用.

我使用python编写代码的时间尚未达到两位数,因此我不能说我对matplotlib的patchs.py有清晰的了解,但我已将解决方案的范围缩小为两种可能的策略:

>聪明的,可能是python的方式:创建一个自定义的arrowstyle类,该类进一步需要对_get_arrow_wedge()函数进行修改以包括中点坐标.目前,这可能超出了我的可能性,或者
>懒惰的方式:从引发的FancyArrowPatch中提取中点坐标,并在此类坐标上绘制所需的箭头样式.

当然,到目前为止,我选择了惰性方式.我做了一些早期的实验,使用get_path()和get_path_in_displaycoord()提取弯曲的FancyArrowPatch的中点坐标,但是我似乎无法预测精确的中点坐标.一些帮助将不胜感激.

到目前为止,我的摆弄:

import matplotlib.pyplot as plt
from matplotlib.patches import FancyArrowPatch

n1 = (2,3)
n2 = (4,6)


# Try with multiple arc radius sizes, draw a separate plot each time
for rad in range(20):
    #setup figure
    figure = plt.figure()
    ax = plt.subplot(111)
    plt.annotate('rad:' + str(rad/25.),xy=(2,5))

    # create rounded fancyarrowpatch   
    t = FancyArrowPatch(posA=n1,posB=n2,
                        connectionstyle='arc3,rad=%s'%float(rad/25.),
                        arrowstyle='->',
                        shrinkA=0,
                        shrinkB=0,
                        mutation_scale=0.5)

    # extract vertices from get_path: points P#
    path = t.get_path().vertices.tolist()
    lab, px, py = ['P{0}'.format(i) for i in range(len(path))], [u[0] for u in path],[u[1] for u in path]
    for i in range(len(path)):
        plt.annotate(lab[i],xy=(px[i],py[i]))

    # extract vertices from get_path_in_displaycoord (but they are useless) : points G#
    newpath = t.get_path_in_displaycoord()
    a,b = newpath[0][0].vertices.tolist(), newpath[0][1].vertices.tolist()
    a.extend(b)
    glab, gx, gy = ['G{0}'.format(i) for i in range(len(a))], [u[0] for u in a],[u[1] for u in a]
    for i in range(len(a)):
        plt.annotate(glab[i],xy=(gx[i],gy[i]))    

    #point A: start
    x1, y1 = n1
    plt.annotate('A',xy=(x1,y1))    

    #point B:end
    x2, y2 = n2
    plt.annotate('B',xy=(x2,y2))

    #point M: the 'midpoint' as defined by class Arc3, specifically its connect() function
    x12, y12 = (x1 + x2) / 2., (y1 + y2) / 2.
    dx, dy = x2 - x1, y2 - y1
    cx, cy = x12 + (rad/100.) * dy, y12 - (rad/100.) * dx    
    plt.annotate('M',xy=(cx,cy))

    #point O : midpoint between M and P1, the second vertex from get_path   
    mx,my = (cx + px[1])/2., (cy + py[1])/2.
    plt.annotate('O',xy=(mx,my))

    ax.add_patch(t)
    plt.scatter([x1,cx,x2,mx,gx].extend(px),[y1,cy,y2,my,gy].extend(py))


plt.show()

python-在matplotlib中用中点箭头绘制圆形的fancyarrowpatch

编辑:上@cphlewis建议:我试图重建贝塞尔曲线:

def bezcurv(start,control,end,tau):
    ans = []
    for t in tau:
        B = [(1-t)**2 * start[i] + 2*(1-t)*t*end[i] + (t**2)*control[i] for i in range(len(start))]
        ans.append(tuple(B))
    return ans

因此,我将生成的线添加到原始图:

tau = [time/100. for time in range(101)]
bezsim = bezcurv(n1,n2,(cx,cy),tau)
simx,simy = [b[0] for b in bezsim], [b[1] for b in bezsim]

下面的绿线是(应该是?)重构的贝塞尔曲线,尽管显然不是.
python-在matplotlib中用中点箭头绘制圆形的fancyarrowpatch

解决方法:

经过很多努力之后,我说服自己,要解决这个问题,我必须放弃FancyArrowPatch套件,并从头开始创建一些东西.这是一个可行的解决方案,远远不满足于任何完美主义的精神,但令我满意:

import matplotlib.pyplot as plt
import numpy as np
from numpy.random import seed, randint

# Build function that connects two points with a curved line, 
# and an arrow on the middle of it

seed(1679)

narrow = 3
rad_one = 50
numpoints = 3

random_points = list(randint(1,20,[numpoints,4]))
rpoints = [[(a,b),(c,d)] for a,b,c,d in random_points]

def curvline(start,end,rad,t=100,arrows=1,push=0.8):
    #Compute midpoint
    rad = rad/100.    
    x1, y1 = start
    x2, y2 = end
    y12 = (y1 + y2) / 2
    dy = (y2 - y1)
    cy = y12 + (rad) * dy
    #Prepare line
    tau = np.linspace(0,1,t)
    xsupport = np.linspace(x1,x2,t)
    ysupport = [(1-i)**2 * y1 + 2*(1-i)*i*cy + (i**2)*y2 for i in tau]
    #Create arrow data    
    arset = list(np.linspace(0,1,arrows+2))
    c = zip([xsupport[int(t*a*push)] for a in arset[1:-1]],
                      [ysupport[int(t*a*push)] for a in arset[1:-1]])
    dt = zip([xsupport[int(t*a*push)+1]-xsupport[int(t*a*push)] for a in arset[1:-1]],
                      [ysupport[int(t*a*push)+1]-ysupport[int(t*a*push)] for a in arset[1:-1]])
    arrowpath = zip(c,dt)
    return xsupport, ysupport, arrowpath

def plotcurv(start,end,rad,t=100,arrows=1,arwidth=.25):
    x, y, c = curvline(start,end,rad,t,arrows)
    plt.plot(x,y,'k-')
    for d,dt in c:
        plt.arrow(d[0],d[1],dt[0],dt[1], shape='full', lw=0, 
                  length_includes_head=False, head_width=arwidth)
    return c

#Create figure
figure = plt.figure()
ax = plt.subplot(111)
for n1,n2 in rpoints:    
    #First line
    plotcurv(n1,n2,rad_one,200,narrow,0.5)
    #Second line
    plotcurv(n2,n1,rad_one,200,narrow,0.5)
ax.set_xlim(0,20)
ax.set_ylim(0,20)
plt.show

我已经用三个随机点对它进行了测试,来回绘制了线条.如下图所示:

python-在matplotlib中用中点箭头绘制圆形的fancyarrowpatch

该功能允许用户设置多个所需的箭头,并将其均匀地放置在绘制的贝塞尔曲线上,以确保显示了正确的方向.但是,由于贝塞尔曲线并不完全是“弧”,因此我试探性地推动箭头的起点以使其看起来更居中.此解决方案的任何改进将不胜感激.

上一篇:linux – 当存在完整路径时,如何使用补丁在不同的文件夹上为文件夹应用diff文件?


下一篇:【patch】使用(DUP) v2.26.1制作patch