Matplotlib: обновить положение патчей (или: set_xy для кругов)

Вдохновленный этот пример Я пытаюсь написать небольшую программу matplotlib, которая позволяет пользователю динамически перетаскивать точки данных в диалоговом режиме, В отличие от примера, в котором используется штриховая диаграмма (и, таким образом, позволяет перетаскивать прямоугольники), моя цель заключалась в том, чтобы добиться того же самого с другими патчами, как, например, круг (любой патч, который более совместим с разбросом, чем прямоугольник, будет). Однако я застрял в обновлении положения моего патча. Пока Rectangle предоставляет функцию set_xy, я не могу найти прямой аналог для Cirlce или Ellipse. Получение положения круга также менее прямолинейно, чем для прямоугольника, но возможно с помощью получения ограничивающего прямоугольника. Отсутствующий кусок теперь должен найти способ обновить положение моего патча. Любой намек на то, как этого достичь, будет замечательным! Текущий минимальный рабочий пример будет выглядеть так:

import numpy as np
import matplotlib.pyplot as plt
import matplotlib.patches as patches

class DraggablePatch:

  def __init__(self, patch):
    self.patch = patch
    self.storedPosition = None
    self.connect()

  def getPosOfPatch(self, marker):
    ext = marker.get_extents().get_points()
    x0 = ext[0,0]
    y0 = ext[0,1]
    x1 = ext[1,0]
    y1 = ext[1,1]
    return 0.5*(x0+x1), 0.5*(y0+y1)

  def connect(self):
    'connect to all the events we need'
    self.cidpress  = self.patch.figure.canvas.mpl_connect('button_press_event',  self.onPress)
    self.cidmotion = self.patch.figure.canvas.mpl_connect('motion_notify_event', self.onMove)

  def onPress(self, event):
    'on button press we will see if the mouse is over us and store some data'
    contains, attrd = self.patch.contains(event)
    if contains:
      self.storedPosition = self.getPosOfPatch(self.patch), event.xdata, event.ydata

  def onMove(self, event):
    'how to update an circle?!'
    contains, attrd = self.patch.contains(event)
    if contains and self.storedPosition is not None:
      oldPos, oldEventXData, oldEventYData = self.storedPosition
      dx = event.xdata - oldEventXData
      dy = event.ydata - oldEventYData
      newX = oldPos[0] + dx
      newY = oldPos[1] + dy
      print "now I would like to move my patch to", newX, newY


def myPatch(x,y): 
  return patches.Circle((x,y), radius=.05, alpha=0.5)

N = 10
x = np.random.random(N)
y = np.random.random(N)
patches = [myPatch(x[i], y[i]) for i in range(N)]

fig = plt.figure()
ax = fig.add_subplot(111)
drs = []
for patch in patches:
  ax.add_patch(patch)
  dr = DraggablePatch(patch)
  drs.append(dr)

plt.show()

Ответ 1

Это немного раздражает, что это противоречиво, но чтобы обновить положение круга, установите circ.center = new_x, new_y.

Как простой (неперетачиваемый) пример:

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

class InteractiveCircle(object):
    def __init__(self):
        self.fig, self.ax = plt.subplots()
        self.ax.axis('equal')

        self.circ = Circle((0.5, 0.5), 0.1)
        self.ax.add_artist(self.circ)
        self.ax.set_title('Click to move the circle')

        self.fig.canvas.mpl_connect('button_press_event', self.on_click)

    def on_click(self, event):
        if event.inaxes is None:
            return
        self.circ.center = event.xdata, event.ydata
        self.fig.canvas.draw()

    def show(self):
        plt.show()


InteractiveCircle().show()