只是为了记录,如果你还是 python 新手,你肯定会直接跳到池子的深处。(感谢你直接进去!)
您正在做的事情需要对 matplotlib 的内部工作有相当详细的了解,这是一个相当复杂的库。
话虽如此,这是一个快速学习的好方法!
对于这样的事情,您需要了解事物结构的内部架构,而不仅仅是“公共”api。
对于大多数情况,您必须深入挖掘并“使用源代码”。对于任何项目,内部工作的文档就是代码本身。
话虽如此,对于一个简单的案例,它非常简单。
import numpy as np
from matplotlib.projections.geo import HammerAxes
import matplotlib.projections as mprojections
from matplotlib.axes import Axes
from matplotlib.patches import Wedge
import matplotlib.spines as mspines
class LowerHammerAxes(HammerAxes):
name = 'lower_hammer'
def cla(self):
HammerAxes.cla(self)
Axes.set_xlim(self, -np.pi, np.pi)
Axes.set_ylim(self, -np.pi / 2.0, 0)
def _gen_axes_patch(self):
return Wedge((0.5, 0.5), 0.5, 180, 360)
def _gen_axes_spines(self):
path = Wedge((0, 0), 1.0, 180, 360).get_path()
spine = mspines.Spine(self, 'circle', path)
spine.set_patch_circle((0.5, 0.5), 0.5)
return {'wedge':spine}
mprojections.register_projection(LowerHammerAxes)
if __name__ == '__main__':
import matplotlib.pyplot as plt
fig = plt.figure()
ax = fig.add_subplot(111, projection='lower_hammer')
ax.grid(True)
plt.show()

让我们深入研究一下这个_get_axes_spines
方法:
def _gen_axes_spines(self):
"""Return the spines for the axes."""
# Make the path for the spines
# We need the path, rather than the patch, thus the "get_path()"
# The path is expected to be centered at 0,0, with radius of 1
# It will be transformed by `Spine` when we initialize it
path = Wedge((0, 0), 1.0, 180, 360).get_path()
# We can fake a "wedge" spine without subclassing `Spine` by initializing
# it as a circular spine with the wedge path.
spine = mspines.Spine(self, 'circle', path)
# This sets some attributes of the patch object. In this particular
# case, what it sets happens to be approriate for our "wedge spine"
spine.set_patch_circle((0.5, 0.5), 0.5)
# Spines in matplotlib are handled in a dict (normally, you'd have top,
# left, right, and bottom, instead of just wedge). The name is arbitrary
return {'wedge':spine}
现在有几个问题:
- 事物没有正确地在轴内居中
- 坐标轴补丁可以放大一点,以适当地占用坐标轴内的空间。
- 我们正在为整个地球绘制网格线,然后剪裁它们。只将它们画在我们的“下”楔子内会更有效。
然而,当我们看一下HammerAxes
它的结构时,你会注意到很多这些东西(尤其是轴补丁的中心)被有效地硬编码到变换中。(正如他们在评论中提到的那样,它是一个“玩具”示例,并且假设您总是在处理整个地球,这使得转换中的数学变得更加简单。)
如果你想解决这些问题,你需要调整HammerAxes._set_lim_and_transforms
.
但是,它按原样工作得相当好,所以我将把它作为练习留给读者。:) (请注意,这部分有点难,因为它需要详细了解 matplotlib 的转换。)