3次元座標変換のメモ書き
これは Zenn で公開している内容と同じです.→ 3次元座標変換のメモ書き
Docusaurusの数式処理を確認するための文書です.ここでは数式表示の処理をクライアント側で行うようにしていますので、Zennの記事で見ると快適だと思います.
はじめに
入社してきた新入社員が期待と不安とやる気に満ちあふれているところで,まずは研修から始めるところが多いかと思います. 研修なんて受けたことがないので,きちんと基礎を叩き込んでくれるところで研修受けたい.そう思うこの頃です.
今回は3次元座標変換について,巷にはそういった情報は沢山あるのですが,自分なりの整理も含めて少し書くことにしました.新人向けに書いていますので,最初は基本的なところから入っていきます.
基本的な座標変換
座標系(coordinate system)
3次元空間は実世界と同じであり,上下・左右・前後の3つの次元です.この上下・左右・前後はそれぞれ直交関係にあります.そして,例えば上下というのは,私から見た上下と,あなたから見た上下は必ずしも一致するとは限りません.つまり,上下・左右・前後というのはどこか基点があるということです.この場合,私やあなたが基点であり,向いている方向に依存しています.次に,例えば前方に5歩進むとしたら人によって進む距離に違いがあります.もし,前方に1メートル進む場合は全員同じ距離になりますが,今度は1メートル進むための歩数が変わってきます.このように,単位も重要になってきます.よって,基点と向きと単位を決めれば,その3次元空間での座標が特定できることになります.この基点を原点(origin)とし,向き(axis),単位(unit)を決めたものを座標系といいます.ここでは上下をY,左右をX,前後をZとします.
3次元空間では座標系を決めるときに2つの種類があります.左手座標系と右手座標系です.左手座標系ではZ方向が奥に向かって正であり,DirectXやUnrealEngine,Unityなどが左手座標系です.右手座標系はZ方向が手前に向かって正であり,OpenGLやWebGLなどが右手座標系 です.座標系が同じだとしても,上下方向がYやZだったりするので,座標系と軸を確認するようにしましょう.
ここでは右手座標系で説明します.
ベクトル(vector)
座標系が決まれば,位置を表現することができます.3次元空間での位置はベクトルで表されます.まず,点$p$の位置をXYZの各軸で測った値をそれぞれx,y,zとすると
$$ p = (x,y,z) $$
で位置を表現できます.ベクトルは方向と大きさを表していて,例えば点$q$から点$p$に向かうベクトル$v$は
$$ q = (0,30,0), \quad p = (0,50,0), \quad \vec{v} = \vec{qp} = (0,20,0) $$
となります.これにはベクトルの始点,終点の位置,大きさ,方向がありますね.ここで,ベクトル$\vec{v}$の始点を原点$O = (0,0,0)$とすれば,
$$ \vec{OV} = \vec{v} = (0,20,0) $$
となって,大きさと方向のみを表すことになります.このときの$x,y,z$のことをそれぞれX成分,Y成分,Z成分といい,この表記を成分表示といいます.
ベクトルの成分表示が出来たので,ベクトルから大きさと方向をそれぞれ取り出してみましょう.まず,大きさですがこれは始点から終点までの距離になります.始点を原点とすればピタゴラスの定理を使ってベクトルの大きさ$|\vec{v}|$は
$$ |\vec{v}| = \sqrt{x^2+y^2+z^2} $$
で得られます.3次元ベクトルが3つの値を持っていることに対して,ベクトルの大きさは1つの量を表す値です.この値をベクトルに対してスカラ(scalar)といいます.
次に方向は,大きさが1のベクトルで表します.このようなベクトルを単位ベクトル(unit vector)といいます.単位ベクトルはベクトルの各成分を大きさで割ることで求められます.
$$ \frac{\vec{v}}{|\vec{v}|} = \left(\frac{x}{|\vec{v}|},\frac{y}{|\vec{v}|},\frac{z}{|\vec{v}|}\right) $$
これを正規化(normalize)といいます.つまり,ベクトルを正規化すればそれは方向を表すベクトルということになります.
※ベクトルや行列の演算について,これから出てきますが詳しいことはここでは解説しません.ベクトルや行列などの線形代数については他を参照したり,もしくは私が書いた「CGのための数学」を参照してください.
基底(basis)
ベクトルの成分表示でXYZで測った値を表記することがわかりました.もちろん,このXYZというのはそのベクトルを測った座標系に基づいています.この座標系とベクトルの関係を見ていきましょう.
XYZというのは座標系における軸のことです.これまで,上下・左右・前後という曖昧な表現でしたが,方向を表すベクトル,つまり単位ベクトルを使うと,XYZの向きをベクトルで表すことが出来ます.一般的な座標系として,左右をX,上下をY,前後をZとしたとき,それぞれの向き(単位ベクトル)$\vec{e} _ x,\vec{e} _ y,\vec{e} _ z$は
$$ \vec{e}_x = (1,0,0), \quad \vec{e}_y = (0,1,0), \quad \vec{e}_z = (0,0,1) $$
となります.これを使うとベクトルの成分表示は
$$ \vec{p} = \left(\begin{matrix}p_x \newline p_y \newline p_z\end{matrix}\right) = p_x \left(\begin{matrix}1 \newline 0 \newline 0\end{matrix}\right) + p_y \left(\begin{matrix}0 \newline 1 \newline 0\end{matrix}\right) + p_z \left(\begin{matrix}0 \newline 0 \newline 1\end{matrix}\right) = p_x \vec{e}_x + p_y \vec{e}_y + p_z \vec{e}_z $$
となります.この$\vec{e}_x,\vec{e}_y,\vec{e}_z$はお互い直交関係になっています.ベクトルの内積が0のとき,直交であることから
$$ \vec{e}_x \cdot \vec{e}_y = \vec{e}_y \cdot \vec{e}_z = \vec{e}_z \cdot \vec{e}_x = 0 $$
の関係であることがわかります.これらの単位ベクトルによって座標系の向きを決めることができます.このXYZの向きベクトルの組を基底といい,それぞれのベクトルのことを基底ベクトルといいます.ベクトルの内積は片方が単位ベクトルのとき,もう片方のベクトルのその単位ベクトル方向の大きさを求められます.これを使って,ある座標系の基底ベクトルと,位置を表すベクトルから
$$ \begin{aligned} p_x &= \vec{p} \cdot \vec{e}_x \newline p_y &= \vec{p} \cdot \vec{e}_y \newline p_z &= \vec{p} \cdot \vec{e}_z \newline \end{aligned} $$
と各成分が得られます.さらに行列を使って,$\vec{e}_x = i, \vec{e}_y = j, \vec{e}_z = k$とおくと
$$ \begin{aligned} M &= \left(\begin{matrix}i_x & i_y & i_z \newline j_x & j_y & j_z \newline k_x & k_y & k_z\end{matrix}\right) \newline \vec{p}' &= M\vec{p} \newline &= \vec{p}\cdot i + \vec{p}\cdot j + \vec{p}\cdot k \end{aligned} $$
となります.この行列$M$はXYZの基底ベクトルで構成されていることから,基底を表 していることが想像できると思います.行列とベクトルの内積の性質から,基底を表す行列$M$に,任意のベクトル$\vec{p}$をかけると(一次変換),行列$M$で表す基底での成分に変換することができ,この値は一意に決まります.よって,この行列$M$はベクトル$\vec{p}$から$\vec{p}'$の写像を表しています.
基底ベクトルというのは常に単位ベクトルというわけでもなく,また,3次元での基底は直交していなくても定義することができます.このような座標系は斜交座標系といったりします.ここでは,右手座標系ですので,基底ベクトルは直交関係であり,また,基底ベクトルが単位ベクトルであるとします.このような基底を正規直交基底といいます.
基底変換(basis transformation)
いよいよ座標変換に入っていきます.座標変換といっても色々種類がありますので,1つずつ見ていきましょう.まずは基底変換です.基底というのはすでに見てきたように,基底ベクトルの組であり,基底ベクトルは単位ベクトルで互いに直交になっています.座標系で出てきた「原点」「向き」「単位」のうち,「原点」は$O=(0,0,0)$のままでどの基底でも同じです.また,単位も基底ベクトルは常に単位ベクトルですので,単位も変わりません.結局,基底は向きのみを表しているといえます.そうすると基底が変わるということはどういうことでしょうか .原点が固定で,大きさも変わらないとなれば,それは向きが「回転」すると考えられます.つまり,基底変換は回転させる座標変換といえます.
ここで少し三角関数について復習しておきましょう.まずは三角比から.
$$ \sin\theta = \frac{a}{c}, \quad \cos\theta = \frac{b}{c}, \quad \tan\theta = \frac{a}{b} $$
単位円で三角関数を見てみると
$$ x = \cos\theta, \quad y = \sin\theta $$
となります.また,ピタゴラスの定理から
$$ \sin\theta^2 + \cos\theta^2 = 1 $$
が得られます.三角関数の逆関数にはそれぞれ$\arcsin, \arccos, \arctan$があります.これらの関係は
$$ \begin{aligned} \theta &= \arcsin(\sin\theta) \quad ( -\frac{\pi}{2} \leqq \theta \leqq \frac{\pi}{2} ), \newline \theta &= \arccos(\cos\theta) \quad ( 0 \leqq \theta \leqq \pi ), \newline \theta &= \arctan(\tan\theta) \quad ( -\frac{\pi}{2} < \theta < \frac{\pi}{2} ) \end{aligned} $$
となります.次に加法定理です.次の図を見てください.
$\Delta abc$において,斜辺の長さが$1$から
$$ \overline{ac} = \sin(A+B), \quad \overline{bc} = \cos(A+B) $$
$\Delta abd$において,
$$ \overline{ad} = \sin B, \quad \overline{bd} = \cos B $$
次に
$$ \begin{aligned} \overline{af} &= \overline{ad}\cos A = \cos A \sin B \newline \overline{fc} &= \overline{de} = \overline{bd}\sin A = \sin A \cos B \end{aligned} $$
$\overline{ac} = \overline{af}+\overline{fc}$の関係から
$$ \begin{aligned} \sin(A+B) &= \sin A\cos B + \cos A\sin B \newline \cos(A+B) &= \cos A\cos B - \sin A\sin B \newline \sin(A-B) &= \sin A\cos B - \cos A\sin B \newline \cos(A-B) &= \cos A\cos B + \sin A\sin B \end{aligned} $$
となります.
それでは回転を見ていきましょう.次の図のように点$p$を点$p'$に回転する場合を考えます.ここでは$z=0$とします.
すると点$p$の各成分は
$$ x = r\cos\phi, \quad y = r\sin\phi $$
となります.回転後の点$p'$の各成分は加法定理を使って
$$ \begin{aligned} x' &= r\cos(\theta+\phi) = r\cos\theta\cos\phi - r\sin\theta\sin\phi \newline y' &= r\sin(\theta+\phi) = r\sin\theta\cos\phi + r\cos\theta\sin\phi \end{aligned} $$
となります.ここで
$$ r = \frac{x}{\cos\phi}, \quad r = \frac{y}{\sin\phi} $$
を代入して,整理すると
$$ \begin{aligned} x' &= \frac{x\cos\theta\cos\phi}{\cos\phi} - \frac{y\sin\theta\sin\phi}{\sin\phi} = x\cos\theta - y\sin\theta \newline y' &= \frac{x\sin\theta\cos\phi}{\cos\phi} + \frac{y\cos\theta\sin\phi}{\sin\phi} = x\sin\theta + y\cos\theta \newline z' &= z \end{aligned} $$
となって,行列で書くと
$$ \left(\begin{matrix}x' \newline y' \newline z'\end{matrix}\right) = \left(\begin{matrix} \cos\theta & -\sin\theta & 0 \newline \sin\theta & \cos\theta & 0 \newline 0 & 0 & 1 \end{matrix}\right) \left(\begin{matrix}x \newline y \newline z\end{matrix}\right) $$
となります.この行列の基底ベクトルに注目してみると,回転させたい方向とは逆の方向 に回転したベクトルになっています.つまり,反対方向に回転させた基底を使って変換を行うと,回転した座標になるということになります.
この回転行列はZ軸を基準に回転を表していることになります.このZ軸回転の行列を$R_z$とおくと
$$ R_z = \left(\begin{matrix} \cos\theta & -\sin\theta & 0 \newline \sin\theta & \cos\theta & 0 \newline 0 & 0 & 1 \end{matrix}\right) $$
となります.同じようにX軸,Y軸の回転行列$R_x,R_y$は
$$ \begin{aligned} R_x &= \left(\begin{matrix} 1 & 0 & 0\newline 0 & \cos\theta & -\sin\theta \newline 0 & \sin\theta & \cos\theta \end{matrix}\right) \newline R_y &= \left(\begin{matrix} \cos\theta & 0 & \sin\theta\newline 0 & 1 & 0 \newline -\sin\theta & 0 & \cos\theta \end{matrix}\right) \end{aligned} $$
となります.ここで各回転行列の行ベクトル,および列ベクトルは
$$ \sin\theta^2 + \cos\theta^2 = 1 $$
から,すべて単位ベクトルであり,列ベクトルにおいても互いに内積がゼロなので直交していることがわかります.
行列は掛け合わせて合成できるので,例えばXYZ回転の行列$R _ {xyz}$は
$$ R _ {xyz} = R_z R_y R_x $$
となります.ここでは取り上げませんが,任意の軸を基準に回転する行列の式も簡単に見つかると思います.これらの回転行列を組み合わせて基底変換の行列をつくることが出来ます.