• Stars
    star
    490
  • Rank 89,811 (Top 2 %)
  • Language
    Python
  • License
    Other
  • Created almost 15 years ago
  • Updated 3 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Python in Maya Done Right

Python in Maya Done Right

PyMEL makes python scripting with Maya work the way it should.

Maya's command module is a direct translation of mel commands into python commands. The result is a very awkward and unpythonic syntax which does not take advantage of python's strengths -- particulary, a flexible, object-oriented design. PyMEL builds on the maya.cmds module by organizing its commands into a class hierarchy, and by customizing them to operate in a more succinct and intuitive way.

Project Goals

  • Create an open-source python module for Maya that is intuitive to MEL users and python users alike
  • Fix bugs and design limitations in Maya's python modues, maya.cmds and maya.mel
  • Keep code concise and readable
  • Add organization through class hierarchy and sub-modules
  • Provide documentation accessible via html and the builtin `help() function
  • Make it "just work"

Supported Maya Versions

PyMEL supports four major versions of Maya:

  • 2020
  • 2022
  • 2023
  • 2024

See the full changelog.

Production Proven

Since its release in 2008, PyMEL has accumulated an impressive resume in both feature films and games, and is now bundled with every release of Maya.

Here's what Seth Gibson of Bungie Studios, makers of Halo, has to say:

Having done production python code myself for many years, wrapping my head around Maya's native implementation took a little bit of time. With PyMel, I can think and write the python code and syntax I'm already used to, which speeds up my development time considerably. It's also going to help our other Technical Artists with their Python learning curve, since PyMEL's syntax is consistent with most other python packages. Kudos to the PyMel team for such a well thought out project!

BSD License

PyMEL is released under the BSD license, which is as open as open source gets. Your studio can freely use, contribute to, and modify this module with no strings attached.

Features

API Hybridization

PyMEL harnesses the API to create a name-independent representation of your object. This means that the annoying inconsistencies of string comparisons are over: no more worrying about short names versus long names, DAG paths, unique paths, instance paths... it's all handled intelligently for you. And what's more, if anything causes the name of your object to change it will automatically be reflected in your python object.

PyMEL node classes now include hundreds of new methods derived from the API, but with the same intuitive and unified design as before. With PyMEL you get the benefits of API speed and versatility without the advanced learning curve.

Improved Batch / Standalone Support

Ever wonder why python scripts that work in the Maya UI or in Maya batch don't work in Maya's python interpreter? Here's a possible explanation: in both GUI and batch modes Maya sources all of its lowest-level MEL scripts, like those that load user plugins, whereas mayapy and maya.initialize does not.

PyMEL ensures that Maya is properly initialized when used from Maya's python interpreter, which makes runtime environments more consistent and your scripts more portable, which adds up to fewer bugs.

Tighter MEL Integration

Executing a MEL script with arguments can be an unsightly mess when done the default way:

default

values = ['one', 'two', 'three', 'four']
maya.mel.eval('stringArrayRemoveDuplicates({"'+'","'.join(values)+'"})')

So PyMEL provides a handy interface which makes calling MEL procedures just like calling a python function:

PyMEL

values = ['one', 'two', 'three', 'four']
pm.mel.stringArrayRemoveDuplicates(values)

Also, unlike maya.mel.eval, PyMEL will give you the specific MEL error message in a python traceback, along with line numbers:

>>> mel.myScript('foo', [])
Traceback (most recent call last):
    ...
MelConversionError: Error occurred during execution of MEL script: line 2: Cannot convert data of type string[] to type float.

Also, getting and setting MEL global variables is as easy as working with a dictionary:

print(pm.melGlobals['gMainFileMenu'])
pm.melGlobals['gGridDisplayGridLinesDefault'] = 2

Powerful Classes

Nodes

camTrans, cam = pm.camera()  # create a new camera
cam.setFocalLength(100)
fov = cam.getHorizontalFieldOfView()
cam.dolly(-3)
cam.track(left=10)
cam.addBookmark('new')

Attributes

s = pm.polySphere()[0]
if s.visibility.isKeyable() and not s.visibility.isLocked():
    s.visibility.set(True)
    s.visibility.lock()
    print(s.visibility.type())

File paths

backup all mb files in the current scene's directory

basedir = pm.sceneName().parent
backupDir = basedir / "backup" #slash op joins paths
if not backupDir.exists:
    backupDir.mkdir()
for path in basedir.files('*.mb'):
    print("backing up: ", path.name)
    path.copy(backupDir / (path.namebase + ".old"))

Shape components and vectors/matrices

select all faces that point up in world space

s = pm.polySphere()[0]
for face in s.faces:
    if face.getNormal('world').y > 0.0:
       pm.select(face, add=1)

optionVars dictionary

if 'numbers' not in pm.optionVar:
    pm.optionVar['numbers'] = [1, 24, 47]
pm.optionVar['numbers'].append(9)
numArray = pm.optionVar.pop('numbers')

Who is PyMEL for?

For the Novice

Object-oriented programming, like that provided by PyMEL, is more intuitive to learn because the functionality of an object is directly associated with the object itself.

For an artist starting to program in Maya, the first question you might ask is "what can I do with this node?" Using a procedural approach, like that offered by MEL or maya.cmds, you'll have to dig through the thousands of MEL commands looking for the one that you want. For a camera node, the camera MEL command is easy to find, but did you find orbit, track, dolly, and tumble, which also work on cameras? What about the API methods?

In PyMEL, all you have to do is type help(nt.Camera) in the python script editor to find out all the things a camera node can do, or just look up the Camera class in the PyMEL docs.

For the MEL Scripter

When we say PyMEL is concise and easy to read, we mean it.

MEL

string $sel[] = `ls -sl`;
string $shapes[] = `listRelatives -s $sel[0]`;
string $conn[] = `listConnections -s 1 -d 0 $shapes[0]`;
setAttr ( $conn[0] + ".radius") 3;

PyMEL

pm.selected()[0].getShape().inputs()[0].radius.set(3)

For the Technical Director

For those looking to master python in a production environment, PyMEL is more than a module for Maya scripting, it is a repository of example python code -- a self-contained pipeline demonstrating advanced python concepts like function factories, metaclasses, and decorators, as well as essential production practices such as parsing, pickling, logging, and unit testing.

For those who are already masters of python and who naturally expect more out of a python package, PyMEL is for you, too. It was written for use in production by experienced programmers with a vision for how to add object-oriented design to Maya.

Code Comparison

with Mel

string $objs[] = `ls -type transform`;
for ($x in $objs) {
    print (longNameOf($x)); print "\n";

    // make and break some connections
    connectAttr( $x + ".sx") ($x + ".sy");
    connectAttr( $x + ".sx") ($x + ".sz");

    // disconnect all connections to .sx
    string $conn[] = `listConnections -s 0 -d 1 -p 1 ($x + ".sx")`;
    for ($inputPlug in $conn)
        disconnectAttr ($x + ".sx") $inputPlug;

    // add and set a string array attribute with the history of this transform's shape
    if ( !`attributeExists "newAt" $x`)
        addAttr -ln newAt -dataType stringArray $x;
    string $shape[] = `listRelatives -s $x`;
    string $history[] = `listHistory $shape[0]`;
    string $elements = "";
    for ($elem in $history)
        $elements += """ + $elem + "" ";
    eval ("setAttr -type stringArray " + $x + ".newAt " + `size $history` + $elements);
    print `getAttr ( $x + ".newAt" )`;

    // get and set some attributes
    setAttr ($x + ".rotate") 1 1 1;
    float $trans[] = `getAttr ($x + ".translate")`;
    float $scale[] = `getAttr ($x + ".scale")`;
    $trans[0] *= $scale[0];
    $trans[1] *= $scale[1];
    $trans[2] *= $scale[2];
    setAttr ($x + ".scale") $trans[0] $trans[1] $trans[2];

    // call a mel procedure
    myMelScript( `nodeType $x`, $trans );
}

with default Python

import maya.cmds as cmds
objs = cmds.ls(type='transform')
# returns None when it finds no matches
if objs is not None:
    for x in objs:
        print(mm.eval('longNameOf("%s")' % x))

        # make and break some connections
        cmds.connectAttr('%s.sx' % x,  '%s.sy' % x)
        cmds.connectAttr('%s.sx' % x,  '%s.sz' % x)

        # disconnect all connections to .sx
        conn = cmds.listConnections(x + ".sx", s=0, d=1, p=1)
        # returns None when it finds no match:
        if conn is not None:
            for inputPlug in conn:
                cmds.disconnectAttr(x + ".sx", inputPlug)

        # add and set a string array attribute with the history of this transform's shape
        if not mm.eval('attributeExists "newAt" "%s"' % x):
            cmds.addAttr(x, ln='newAt', dataType='stringArray')
        shape = cmds.listRelatives(x, s=1 )
        if shape is not None:
            history = cmds.listHistory( shape[0] )
        else:
            history = []
        args = tuple(['%s.newAt' % x, len(history)] + history)
        cmds.setAttr(*args,  type='stringArray' )

        # get and set some attributes
        cmds.setAttr('%s.rotate' % x, 1, 1, 1)
        scale = cmds.getAttr('%s.scale' % x)
        # maya packs the previous result in a list for no apparent reason:
        scale = scale[0]
        # the tuple must be converted to a list for item assignment:
        trans = list(cmds.getAttr('%s.translate' % x )[0])  
        trans[0] *= scale[0]
        trans[1] *= scale[1]
        trans[2] *= scale[2]
        cmds.setAttr('%s.scale' % x, trans[0], trans[1], trans[2])
        # call a mel procedure
        mm.eval('myMelScript("%s",{%s,%s,%s})' % (cmds.nodeType(x), trans[0], trans[1], trans[2]))

with Pymel

# safe to import into main namespace (but only recommended when scripting interactively)
from pymel import *
for x in ls(type='transform'):
    # object oriented design
    print(x.longName())

    # make and break some connections
    x.sx.connect(x.sy)
    x.sx.connect(x.sz)
    # disconnect all connections to .sx
    x.sx.disconnect()

    # add and set a string array attribute with the history of this transform's shape
    x.setAttr('newAt', x.getShape().history(), force=1)

    # get and set some attributes
    x.rotate.set([1, 1, 1])
    trans = x.translate.get()
    # vector math:
    trans *= x.scale.get()
    # ability to pass list/vector args
    x.translate.set(trans)
    # call a mel procedure
    mel.myMelScript(x.type(), trans)

With PyMEL, python and maya play well together.

Installation

To install, use pip with the mayapy interpreter.

Installing into Maya's site-packages directory

The following commands will install PyMEL into the site-packages of the Maya application directory:

On MacOS:

/Applications/Autodesk/maya<Version>/Maya.app/Contents/bin/mayapy -m pip install pymel

On Linux:

sudo /usr/autodesk/maya<Version>/bin/mayapy -m pip install pymel

On Windows:

(from a command window running as Administrator)

/Applications/Autodesk/maya<Version>/Maya.app/Contents/bin/mayapy -m pip install pymel

Installing into your user's site-packages directory

If you do not have permissions to install into Maya's site-packages, you can install into your user site-packages by adding the --user flag, e.g.

/usr/autodesk/maya<Version>/bin/mayapy -m pip install pymel --user

To see where pymel was installed you can run:

mayapy -c "import pymel;print(pymel.__file__)"

Caveats

Older versions of Maya (2021 and earlier) may require installing pip before running pip install:

On Linux and Mac:

curl https://bootstrap.pypa.io/get-pip.py | /usr/autodesk/maya2021/bin/mayapy

PyMEL is developed and maintained by Luma Pictures.

More Repositories

1

usd-qt

Reusable Qt Components for Pixar's USD
Python
153
star
2

cg-stubs

Python stubs for VFX and Animation
Python
141
star
3

LumaNukeGizmos

Gizmos for nuke
Python
129
star
4

meteor-jquery-datatables

Sort, page, and filter millions of records reactively.
CoffeeScript
98
star
5

usd-arnold

USD schemas, exporters, and importers for Arnold renderer
C++
66
star
6

usd-uri-resolver

Pluggable URI ArResolver for USD
CMake
48
star
7

openvdb-render

Toolset for visualizing and rendering openvdb volumes in Maya
C++
41
star
8

pflow

Flow based programming for Python
Python
27
star
9

meteor-jquery-select2

A Blaze UI select2 component that supports reactive search.
CoffeeScript
26
star
10

arnold-rv

Driver for sending arnold frame buffers to Tweak's RV (Righteous Viewer).
C++
24
star
11

houdini-engine-clarisse

Houdini Engine integration for Isotropix Clarisse
C++
24
star
12

meteor-luma-ui

Routes, Styles, Components, and much more
HTML
23
star
13

gozbruh

Sync meshes between Maya and ZBrush running on different workstations
Python
22
star
14

nukecli

Nuke Command-line Tools
Python
17
star
15

setpkg

a command-line utility for managing environment variables
Python
12
star
16

usd-qt-components

Reusable UI components for viewing and authoring USD files
Python
12
star
17

bbcp2

bbcp - Fast file transfer over WAN
C++
10
star
18

luma_usd

Plugins for USD
C++
10
star
19

maya-to-hydra

This repo is no longer updated. Please see https://github.com/Autodesk/maya-usd
C++
10
star
20

meteor-jquery-sparklines

A reactive blaze component for rendering live jQuery Sparklines.
CoffeeScript
9
star
21

rez-build

7
star
22

meteor-spiner

A reactive REST API that builds structured JSON from PostgreSQL Records
CoffeeScript
4
star
23

pymel-docs

pymel documentation for readthedocs
Python
4
star
24

rt-slam

Luma fork of the rt-slam project, from LAAS
C++
4
star
25

qt_tools

Source code for the qt_tools project by David Van Brink
C
3
star
26

maya-usd-scripts

Scripts for helping with merging issues for Maya-USD and "old" Pixar and Animal Logic Repos
Shell
3
star
27

dbdump

Simple unidirectional MySQL data synchronization without complex configuration
Python
2
star
28

robotpkg

Luma fork of robotpkg project, from LAAS
C
2
star
29

meteor-luma-component

Blaze UI Component Mixins
CoffeeScript
1
star
30

ftrack_performance_tests

Comparison of Ftrack API to Sqlalchemy and raw SQL
Python
1
star
31

meteor-luma-admin

CoffeeScript
1
star
32

openrobot-MTI

luma fork of the openrobots MTI-interface library
C++
1
star
33

gdoc-example

Google Docs Example
CSS
1
star
34

meteor-backbone-relational

Client and Server models for handling relational data.
JavaScript
1
star
35

meteor-jquery-caret

Manipulate the caret's position in a text box or content-editable element
JavaScript
1
star
36

jafar

Luma fork of jafar project, from LAAS
C++
1
star