matplotlib 如何优化zoomed_inset_axes中的缩放参数?

k2fxgqgv  于 2022-12-23  发布在  其他
关注(0)|答案(1)|浏览(259)

我正在创建包含缩放插入的绘图。数据是多种多样的,我不可能在程序启动之前知道数据的样子。我想使缩放插入尽可能地放大,而不与绘图的任何其他元素重叠。下面是一个示例,我使用的缩放为2。理想情况下,我想自动确定该数字应该是多少:

  1. import numpy as np
  2. import matplotlib.pyplot as plt
  3. from mpl_toolkits.axes_grid1.inset_locator import zoomed_inset_axes
  4. from mpl_toolkits.axes_grid1.inset_locator import mark_inset
  5. fig, ax = plt.subplots()
  6. xin = np.linspace(0, np.random.uniform(.5, 4), 1000)
  7. x_samples = np.random.uniform(0.9, 1.1, (1, 1000)) * np.sqrt(xin[:, np.newaxis])
  8. ax.fill_between(xin, x_samples.min(1), x_samples.max(1))
  9. axins = zoomed_inset_axes(ax, zoom=2, loc='upper left')
  10. axins.fill_between(xin, x_samples.min(1), x_samples.max(1))
  11. axins.set_xlim(.05, .1)
  12. idx = np.logical_and(xin > 0.05, xin < 0.1)
  13. axins.set_ylim(x_samples.min(1)[idx].min(), x_samples.max(1)[idx].max())
  14. axins.set_xticks([])
  15. axins.set_yticks([])
  16. mark_inset(ax, axins, loc1=4, loc2=3, fc="none", ec="0.5")
  17. plt.savefig('hey')
  18. plt.clf()

如您所见,zoom=2的值太小。我可以手动将缩放参数设置为正确的值。这是一个繁琐的过程。是否有一种方法可以自动找到缩放参数,使插入尺寸最大化,同时避免与绘图的其他部分重叠?

hzbexzde

hzbexzde1#

我们可以用迭代的方式来面对这个问题:
1.从最大可能的缩放开始(使插图占据图的整个高度)。因此,插图的一部分将与图重叠。
1.检查重叠开始点之前存在多少垂直间隙。
1.根据插图的“当前”高度,将其缩小以避免重叠。
1.在重新缩放之后,插图的宽度也会减小,因此我们可以使用留下的自由垂直间隙再次将其放大。
1.返回到2.,直到达到收敛/最大迭代次数。
在实践中,收敛是快速的,并且对于给定的数据在少于10次迭代中达到。
目视检查:

左上角 * 位置处的插图的代码**。

  1. import numpy as np
  2. import matplotlib.pyplot as plt
  3. from mpl_toolkits.axes_grid1.inset_locator import zoomed_inset_axes, mark_inset
  4. def get_inset_max_zoom(x, y, xlim_inset, ylim_inset, max_iters=10):
  5. """ Zoom that maximizes inset size without overlapping the artists """
  6. # width and height of the inset in non-scaled coordinates.
  7. inset_w = xlim_inset[1] - xlim_inset[0]
  8. inset_h = ylim_inset[1] - ylim_inset[0]
  9. # max y-coordinate of the whole plot.
  10. y_max_plot = y.max()
  11. # start with maximum zoom.
  12. y_gap = y_max_plot - y.min()
  13. zoom = y_gap / inset_h
  14. for i in range(max_iters):
  15. y_affected_max = y[x < zoom * inset_w].max()
  16. # recalculate zoom by adjusting the gap.
  17. y_gap = y_max_plot - y_affected_max
  18. zoom = y_gap / inset_h
  19. return zoom
  20. if __name__ == "__main__":
  21. # Change the seed to show produce different values.
  22. rng = np.random.RandomState(seed=0)
  23. # main plot.
  24. fig, ax = plt.subplots()
  25. xin = np.linspace(0, rng.uniform(.5, 4), 1000)
  26. x_samples = rng.uniform(
  27. 0.9, 1.1, (1, 1000)) * np.sqrt(xin[:, np.newaxis])
  28. ax.fill_between(xin, x_samples.min(1), x_samples.max(1))
  29. # get xy pairs.
  30. y = x_samples.ravel()
  31. x = np.repeat(xin, x_samples.shape[1])
  32. # define the limits and location of the zoom inset.
  33. xlim_inset = (.05, .1)
  34. idx = np.logical_and(xin > xlim_inset[0], xin < xlim_inset[1])
  35. ylim_inset = (x_samples.min(1)[idx].min(), x_samples.max(1)[idx].max())
  36. loc = 'upper left'
  37. # get max zoom.
  38. zoom = get_inset_max_zoom(x, y, xlim_inset, ylim_inset, max_iters=5)
  39. # create the inset.
  40. axins = zoomed_inset_axes(ax, zoom=zoom, loc=loc, borderpad=0.5)
  41. axins.set(
  42. xlim=xlim_inset,
  43. ylim=ylim_inset,
  44. xticks=[], yticks=[])
  45. # connect the bboxes.
  46. mark_inset(ax, axins, loc1=4, loc2=3, fc="none", ec="0.5")
  47. # plot within the inset.
  48. axins.fill_between(xin, x_samples.min(1), x_samples.max(1))

泛化到4个角位置{* 左上角 右上角 右下角 左下角 *}。例如,对于loc = 'lower right'

  1. import numpy as np
  2. import matplotlib.pyplot as plt
  3. from mpl_toolkits.axes_grid1.inset_locator import zoomed_inset_axes, mark_inset
  4. def get_inset_max_zoom_given_loc(
  5. x, y, xlim_inset, ylim_inset, loc='upper left', max_iters=10):
  6. """ Zoom that maximizes inset size without overlapping the artists """
  7. # width and height of the inset in non-scaled coordinates.
  8. inset_w = xlim_inset[1] - xlim_inset[0]
  9. inset_h = ylim_inset[1] - ylim_inset[0]
  10. # handy variables.
  11. is_left = 'left' in loc
  12. is_upper = 'upper' in loc
  13. y_min_plot, y_max_plot = y.min(), y.max()
  14. y_xtm_plot = y_max_plot if is_upper else y_min_plot
  15. x_max_plot = x.max()
  16. # start with maximum zoom.
  17. y_gap = y_max_plot - y_min_plot
  18. zoom = y_gap / inset_h
  19. for i in range(max_iters):
  20. # get affected x-coordinate range.
  21. if is_left:
  22. x_affected = x < zoom * inset_w
  23. else:
  24. x_affected = x > x_max_plot - zoom * inset_w
  25. # get affected y-coordinate extremum.
  26. y_affected = y[x_affected]
  27. y_affected_xtm = y_affected.max() if is_upper else y_affected.min()
  28. # recalculate zoom by adjusting the gap.
  29. y_gap = abs(y_xtm_plot - y_affected_xtm)
  30. zoom = y_gap / inset_h
  31. return zoom
  32. if __name__ == "__main__":
  33. # Change the seed to show produce different values.
  34. rng = np.random.RandomState(seed=0)
  35. # main plot.
  36. fig, ax = plt.subplots()
  37. xin = np.linspace(0, rng.uniform(.5, 4), 1000)
  38. x_samples = rng.uniform(
  39. 0.9, 1.1, (1, 1000)) * np.sqrt(xin[:, np.newaxis])
  40. ax.fill_between(xin, x_samples.min(1), x_samples.max(1))
  41. # get xy pairs.
  42. y = x_samples.ravel()
  43. x = np.repeat(xin, x_samples.shape[1])
  44. # define the limits and location of the zoom inset.
  45. xlim_inset = (.05, .1)
  46. idx = np.logical_and(xin > xlim_inset[0], xin < xlim_inset[1])
  47. ylim_inset = (x_samples.min(1)[idx].min(), x_samples.max(1)[idx].max())
  48. loc = 'lower right'
  49. # get max zoom.
  50. zoom = get_inset_max_zoom_given_loc(
  51. x, y, xlim_inset, ylim_inset, loc=loc, max_iters=10)
  52. # create the inset.
  53. axins = zoomed_inset_axes(ax, zoom=zoom, loc=loc, borderpad=0.5)
  54. axins.set(
  55. xlim=xlim_inset,
  56. ylim=ylim_inset,
  57. xticks=[], yticks=[])
  58. # connect the bboxes.
  59. mark_inset(ax, axins, loc1=4, loc2=3, fc="none", ec="0.5")
  60. # plot within the inset.
  61. axins.fill_between(xin, x_samples.min(1), x_samples.max(1))
展开查看全部

相关问题