4
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

お手軽スクショ Maya SSSS(super screen shot system)

Last updated at Posted at 2021-12-17

Maya Advent Calendar 2021

一年ぶりでございます。もはや恒例になりました。

昨年はさくっとアドカレ埋まったのですが、今年は埋まりが悪いようですね。

今回も、今年調査して自分的ホットになったネタを紹介していきます。

今年はスクリーンショットでございます。

スーパーって書いてるけど大したことはないです。ただ、Sを重ねたかっただけです( ´_ゝ`)





この記事でのMayaは全部2020でやってます。2022では確認していないのでそこはご了承ください。

使用用途

さて、そもそもどんなときにスクリーンショットを使うのかというところから。

基本的にはデータのサムネとして使いたいときに行います。

例えば、アセットブラウザ。

UE4でのサムネ表示みたいな感じですね。

advcale_mayapy_2021_01.png

shotgrid(名前変わってややこしい)でもアセットの登録時に使えることでしょう。

ちょっとした変わり種だと、リグのpickerの背景というのもあります。

advcale_mayapy_2021_02.png

正直なところなくても代用が効くけどあるとそれっぽくなるってやつですね。

前の記事とかでも言ってるので重複になりますけど、

ツールは「アフォーダンスがある」、「情報がわかりやすい」と受け入れやすいのでUIにも凝る意味があるんですよね。





Mayaでのスクリーンショット

さて、前置きはこんなもので。Mayaでのスクリーンショットを取る方法を見ていきましょう。

まず、プログラム的な知識がない場合、レンダリングすることが思い浮かぶのではないかと思います。

方法としては全然ありです。私も昔考えるのめんどかったときはバッチでレンダリング回して作成してました。

それ以外での方法はコマンドによるものです。

レンダリングも究極的には全部スクリプトで自動化できますが、

設定がめんどいのとコマンドでサクッと回したいので今回はコマンドによるスクリーンショットにフォーカスを当てます。





mayaPython

from maya import cmds
cmds.modelEditor(panel, e=True, cpt=save_path)

or

cmds.refresh(fe='png', fn=save_path)

save_pathは任意の保存先のパス、panelはモデルパネル名になります。

「っえ!二つあるの!?」とびっくりしました。

というか、普通にスクリーショットっぽいコマンド名で検索しても引っ掛からないので、

そもそも存在してないと思った方もいるんじゃないかと思います。

かくいう自分も方々探しまくったのと、偶然に他の目的で引いたコマンドリファレンスにフラグが存在したのを見つけた次第です。

Autodeskさん不親切…

ちょっと脱線しましたが一個ずつ解説します。

cmds.modelEditor(panel, e=True, cpt=save_path)

capture(cpt)がスクショのフラグです。

フラグの名前だけみればわかりやすいんですが、cmds.modelEditorのフラグは↓こんだけあります。

advcale_mayapy_2021_03.png

わかんないって!

次。

cmds.refresh(fe='png', fn=save_path)

こちらはビューの描画をリフレッシュするときのコマンドになります。

このコマンドには、filename(fn)という引数が存在しています。

リファレンス曰く、

Specify the name of a file in which to save a snapshot of the viewports, or just the current one if the currentView flag is set.
ビューポートのスナップ ショットを保存するファイルの名前を指定します。currentView フラグが設定されている場合は、現在の 1 つを指定します。

これは気づかないですね…

ちなみに、fileExtension(fe)でpngを指定していますが、jpgとかgifとかもできます。

さらには、fileExtension(fe)を指定しなくともパスで拡張子をちゃんと記述しておけばそちらで認識されます。

対象のビューですが、直前までアクティブだったものが対象になります。

なので、コマンドを叩く前に撮りたい対象のビューをアクティブにするコマンドを叩くのを忘れないように。

cmds.setFocus(panel)

つまるところ、

cmds.setFocus(panel)
cmds.refresh(fn=save_path)

と書けば、cmds.modelEditor側と同じになるということです。

openMaya

openMayaで実行する方法もあるので一応ご紹介。

[Maya] Viewport をキャプチャする

もってぃさんありがとう!

もってぃさん元気でやってるかな…閑話休題

余談:panel名の取得について

cmdsm.getPanel(wf=True)

これでアクティブなものをとれます。

ただし、scriptEditorとかでこれを実行すると、必然的にアクティブになっているのがscriptEditorのパネルになってしまうので注意です。

カメラから取得したいときはこれ。

# ---------------------------------------------------------
# get showing modelPanel name
# ---------------------------------------------------------
# @param <str>cam : cam name
# @return <obj>activePanel : showing model panel
def get_viewpanel(cam='persp'):
    panels = cmds.getPanel(typ='modelPanel')
    if panels:
        active_panel = None
        for pnl in panels:
            panel_cam = cmds.modelPanel(pnl, q=True, cam=True)
            if panel_cam == cam:
                active_panel = pnl
                break
    else:
        active_panel = None
    return active_panel

引数camにカメラ名を入れるとそのカメラで映しているpanel名が取れます。





撮ったけど…

さて、とりあえずこれで撮れることがわかりました。

ところがどっこい、この方法でpng出力すると…

advcale_mayapy_2021_04.png

こんな感じで背景が透過してしまいます。サムネとしては見辛い…

もとはこんな感じ

advcale_mayapy_2021_04.jpg

「いやでも容量的に画質的にもpngの方がいいし、pngで撮りたい!」

またしても痒い所に手が届かないところで悩まされます(’A`)

というわけで、実はこの後からが本当の本題になります。





pysideでのスクリーンショット

はい。というわけでまた困ったときのpysideでございます。

ご明察のとおり、pysideでもスクリーンショットを撮ることができます。

from PySide2 import QtGui
ss = QtGui.QPixmap.grabWindow(widget.winId())
ss.save(save_path)

widgetはpysideのQtWidgets.QWidgetです。

panelをどのようにQtWidgets.QWidgetに変換するかはMaya UIにひとてま この記事で解決してるので割愛します。

QtGui.QPixmap.grabWindow関数を使ってwidgetのスクリーンショットをメモリに保存します。

QtGui.QPixmapオブジェクトになっていますのでついでに大きさも調整してしまいましょう。

あとはそれをファイルに出力することで完了です。

汎用的にするとこんな感じ。

import maya.OpenMayaUI as omui
from PySide2 import QtGui
from PySide2 import QtWidgets
try:
    import shiboken2
except ImportError:
    from PySide2 import shiboken2
# ---------------------------------------------------------
# take screenshot by maya UI pointer
# ---------------------------------------------------------
# @param <str>ui_str     : maya UI name
# @param <str>save_path  : save path
# @param <int>height     : image height
# @return <bool>result : result 
def take_ss_in_maya_ui(ui_str, save_path, height):
    tmp_ptr = omui.MQtUtil.findControl(ui_str)
    if not tmp_ptr:
        tmp_ptr = omui.MQtUtil.findLayout()
    if not tmp_ptr:
        tmp_ptr = omui.MQtUtil.findWindow()
    ui_ptr = tmp_ptr
    if not ui_ptr:
        return False

    widget = shiboken2.wrapInstance(long(ui_ptr), QtWidgets.QWidget)
    ss = QtGui.QPixmap.grabWindow(widget.winId())
    ss = ss.scaledToHeight(height)
    try:
        ss.save(save_path)
        result = True
    except Exception:
        result = False
    return result

.e.g.

take_ss_in_maya_ui('modelPanel4', 'D:/hoge/ss.png', 512)

exceptのとこはもっとエラー振り分けた方がいいんですが、撮るだけならこれでまぁいいかなと。

ui_strにはpanel名を入力します。

heightで高さの解像度を指定して、縦横比率そのままにすることで幅も決定してます。

逆にしてもいいですし、両方指定できるようにしてもいいのでそこはご自由にということで。

ちなみに、これで撮ったものはmayaのコマンドで撮ったものと若干違います。

advcale_mayapy_2021_04.jpgadvcale_mayapy_2021_05.png

上がmayaのコマンド、下がpysideです。

若干panelの範囲が違っているのがわかると思います。

個人的にはUIも微妙に映っている方がスクショっぽいのでいいかなとも思うのですが、そこは用途次第ですかね。



さてさて、これの利点はpanelのスクショだけでなく、mayaのありとあらゆるウィンドウをスクショできるところにあります。

ui_strにmaya内のウィンドウ名を入れるだけ。あら簡単。

これ一個書いとけば万々歳というわけですね。やっぱpysideステキ。





おわに

以上がスクリーンショットの方法になります。

ビューをただスクリーンショットすることから始まったのに、

結果的にスーパー汎用的なスクショ関数が出来てしまったのが個人的に激熱でした。

こんな感じで妥協しないで突き詰めていった結果、想定を超えるものができると脳汁ドバドバするのでオススメです。








ということでここまで。よいお年を(:3ノシ )ヘ



* ※免責事項※
本記事内で公開しているコードの安全性について、当方は一切の保証を与えておりません。
これらのコードを使用したことによって引き起こる損害に対し、当方は一切責任を負うものではありません。
自己責任でご使用ください。*

4
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?