18

Python で Matplotlib を使用して 3D サーフェス プロットを行っていますが、厄介な現象に気付きました。視点 (カメラの位置) の設定方法に応じて、垂直 (z) 軸が左右の間を移動します。以下に 2 つの例を示します。例 1、左軸例 2、右軸。最初の例には ax.view_init(25,-135) があり、2 番目の例には ax.view_init(25,-45) があります。

視点を同じに保ちたい (データを表示する最良の方法)。軸を一方または他方に強制する方法はありますか?

4

2 に答える 2

15

似たようなものが必要でした: 両側に zaxis を描きます。@crayzeewulf の回答のおかげで、次の回避策にたどり着きました(左側、右側、または両側)。

ここに画像の説明を入力

最初に必要に応じて 3D をプロットしてから、show()wrapを呼び出す前にAxes3D、単にメソッドをオーバーライドする Wrapper クラスを使用しますdraw()

ラッパー クラス呼び出しは、一部の機能の可視性を False に設定するだけで、自分自身を描画し、最後に変更された PLANES で z 軸を描画します。このラッパー クラスを使用すると、z 軸を左側、右側、または両側に描画できます。

import matplotlib
matplotlib.use('QT4Agg')
import matplotlib.pyplot as plt
from mpl_toolkits.mplot3d import axes3d

class MyAxes3D(axes3d.Axes3D):

    def __init__(self, baseObject, sides_to_draw):
        self.__class__ = type(baseObject.__class__.__name__,
                              (self.__class__, baseObject.__class__),
                              {})
        self.__dict__ = baseObject.__dict__
        self.sides_to_draw = list(sides_to_draw)
        self.mouse_init()

    def set_some_features_visibility(self, visible):
        for t in self.w_zaxis.get_ticklines() + self.w_zaxis.get_ticklabels():
            t.set_visible(visible)
        self.w_zaxis.line.set_visible(visible)
        self.w_zaxis.pane.set_visible(visible)
        self.w_zaxis.label.set_visible(visible)

    def draw(self, renderer):
        # set visibility of some features False 
        self.set_some_features_visibility(False)
        # draw the axes
        super(MyAxes3D, self).draw(renderer)
        # set visibility of some features True. 
        # This could be adapted to set your features to desired visibility, 
        # e.g. storing the previous values and restoring the values
        self.set_some_features_visibility(True)

        zaxis = self.zaxis
        draw_grid_old = zaxis.axes._draw_grid
        # disable draw grid
        zaxis.axes._draw_grid = False

        tmp_planes = zaxis._PLANES

        if 'l' in self.sides_to_draw :
            # draw zaxis on the left side
            zaxis._PLANES = (tmp_planes[2], tmp_planes[3],
                             tmp_planes[0], tmp_planes[1],
                             tmp_planes[4], tmp_planes[5])
            zaxis.draw(renderer)
        if 'r' in self.sides_to_draw :
            # draw zaxis on the right side
            zaxis._PLANES = (tmp_planes[3], tmp_planes[2], 
                             tmp_planes[1], tmp_planes[0], 
                             tmp_planes[4], tmp_planes[5])
            zaxis.draw(renderer)

        zaxis._PLANES = tmp_planes

        # disable draw grid
        zaxis.axes._draw_grid = draw_grid_old

def example_surface(ax):
    """ draw an example surface. code borrowed from http://matplotlib.org/examples/mplot3d/surface3d_demo.html """
    from matplotlib import cm
    import numpy as np
    X = np.arange(-5, 5, 0.25)
    Y = np.arange(-5, 5, 0.25)
    X, Y = np.meshgrid(X, Y)
    R = np.sqrt(X**2 + Y**2)
    Z = np.sin(R)
    surf = ax.plot_surface(X, Y, Z, rstride=1, cstride=1, cmap=cm.coolwarm, linewidth=0, antialiased=False)

if __name__ == '__main__':
    fig = plt.figure(figsize=(15, 5))
    ax = fig.add_subplot(131, projection='3d')
    ax.set_title('z-axis left side')
    ax = fig.add_axes(MyAxes3D(ax, 'l'))
    example_surface(ax) # draw an example surface
    ax = fig.add_subplot(132, projection='3d')
    ax.set_title('z-axis both sides')
    ax = fig.add_axes(MyAxes3D(ax, 'lr'))
    example_surface(ax) # draw an example surface
    ax = fig.add_subplot(133, projection='3d')
    ax.set_title('z-axis right side')
    ax = fig.add_axes(MyAxes3D(ax, 'r'))
    example_surface(ax) # draw an example surface
    plt.show()
于 2014-08-01T15:15:22.733 に答える
11

OPによる以下のコメントで指摘されているように、以下に提案された方法は、元の質問に対する適切な回答を提供しませんでした。

このメモで述べたように、 axis3dには多くのハードコードされた値があり、その動作をカスタマイズするのが困難です。したがって、現在のAPIでこれを行うための良い方法はないと思います。以下に示すようにの_PLANESパラメータを変更することにより、それを「ハック」することができます。zaxis

tmp_planes = ax.zaxis._PLANES 
ax.zaxis._PLANES = ( tmp_planes[2], tmp_planes[3], 
                     tmp_planes[0], tmp_planes[1], 
                     tmp_planes[4], tmp_planes[5])
view_1 = (25, -135)
view_2 = (25, -45)
init_view = view_2
ax.view_init(*init_view)

これで、図をどのように回転させても(正のz方向が上を向いている限り)、z軸は常に図の左側になります。ただし、x軸とy軸は反転し続けます。で遊ぶこと_PLANESができ、すべての軸で目的の動作を取得できる可能性がありますが、これはの将来のバージョンで機能しなくなる可能性がありmatplotlibます。

于 2013-02-24T04:24:15.067 に答える