"""
Name-US: df_pyApplyTex_v0.1(v15) 12-03-2017
Description-US: Places textures on all objects in hierarchy baed on object->material matching names.
Author: David Flamholc, vfxvoodoo.com
# pyApplyTex_version
pyApply = "df_pyApplyTex_v0.1(v15) 12-03-2017"
LICENSE
-------
Copyright (c) 2016, David Flamholc, VFXVoodoo.com
Programming: David Flamholc <dflamholc@gmail.com>
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
CHANGE LOG
----------
version: paApplyTex_v08
- implemented UNDO functions
version: paApplyTex_v09
- implemented moving newly created tags to last in the stack
version: paApplyTex_v10 -- (df_pyApplyTex_v0.1(v10) 12-03-2017)
- refactoring of funcitons and variables
- and FLOW comments
- scrip header INFO and TO DO list added
# THE FLOW ######
# in the MAIN function we call the recurseObjs function
# recurseObjs iterates through the hierarchy and calls the checkName function on every object
# checkName checks the objects name against the names in the material list and runs placeTags on matchin objects
# gets the material matching the object name from the list of materials and sets the tags projection type,
# then sets the material in the empty textag that was created, sets teh projection
KNOWN BUGS
------------------------
- object that already have a non matching material on them will keep stacking new matching materials endlessly
-
INSTRUCTIONS & DEV NOTES
- objects and materials need to have exactly the same name for this script to work
- This Software should work on all os platforms.
- It has only been tested on Cinema 4D r18.
TO DO -- FEATURE IDEAS
-----------------------
[x] Write script comments
[] Create Icon
[] Write Usage Instructions
[] Convert to Plugin
[] Get Unique ID
[] check PART OF NAME so that matching can be done by _TOKEN only on BOTH ON Material or Object
[] fix material stacking BUG on object that already have a non matching tex tag applied (NON MATCHING?? how)
[] ADD feature to control mapping of Tex tag X & LENGTH from text file
Length U == Texture[c4d.TEXTURETAG.LENGTHX] & Length V == Texture[c4d.TEXTURETAG.LENGTHY]
these are presented in 0%-100% but work in normalised float values 0-1
[] cover for None Name TEX Tags
[] if the object has only one non-matching tag then new correct tag is added,
if teh object has two non matching tags then no new tag is added - LIMIT OF TWO MATERIALS
--->> BUT unable to reproduce the error of infinite stacking of tags
# ====== MODULE IMPORT ======#
import c4d
import os
# FUNCTIONS ##############################################################################################
def recurseObjs(obj, checkName):
while obj:
checkName(obj)
recurseObjs(obj.GetDown(), checkName)
obj = obj.GetNext()
return obj
def checkName(obj):
objName = obj.GetName()
matList = doc.GetMaterials()
matNames = [mat.GetName() for mat in matList]
# check for TOKEN in both MATERIAL and OBJECT NAME
if objName in matNames:
checkTags(obj, objName, matList, matNames)
else:
print "No matching names or name _tokens!"
def checkTags(obj, objName, matList, matNames):
objTEX = obj.GetTag(c4d.Ttexture)
objUVW = obj.GetTag(c4d.Tuvw)
tag = obj.GetFirstTag()
tags = obj.GetTags()
proj = None
# check if there is a TEX tag on the object but no UVW tag
if objTEX and not objUVW in tags:
while tag:
if tag == objTEX:
checkTag.append(tag)
if not tag.GetNext():
break
tag = tag.GetNext()
# check the new temp list for name of tex tags
for tTag in checkTag:
if tTag.GetMaterial().GetName() == objName:
return False # do nothing
proj = 'cubic'
placeTag(obj, objName, matList, matNames, proj) # call tag placement funcitons with CUBIC
elif objTEX and objUVW in tags:
# checkTag collects the name of each material tag only
checkTag = [t.GetMaterial().GetName() for i, t in enumerate(tags) if t.GetType() == 5616]
# checking if matching tag is in checkTag and if a matching tag is at the end
if objName in checkTag and objName == checkTag[-1]:
print "we need to move all matching tags that are not at the end, to the end"
print checkTag[:-1]
matchingTags = [t for t in checkTag[:-1] if t == objName]
# the amount of iterations == len(matchingTags)
# if a matching tag were to be in the middle we need to move it second to last.
for i, t in enumerate(checkTag[:-1]):
# the tags order will change as we move the first one
# can we control it with a count ?
count = 0
if t == objName and count >= 0:
print i
print "this is the tag to move to position [-2]"
count += 1
elif count < 0:
print "this tag has to move to position [-3]"
# remove tags - but can't do this based on name
# maybe above instead of creating a list of names instead remove tags based on name
# and then reinsert them
# also check for selection tags
elif objName in checkTag and objName != checkTag[-1]:
print "we'll move the tag to the end"
# tagPos = [m for m in checkTag if objName in checkTag]
moveTag = checkTag.index(objName)
# we need to find the index of the matchin tag in the list tags
pass
# move tag to last tag
# get position of tag
# check what projection in checkTag[-1]
proj = 'uvw'
# if statement to see if the objname appears now
placeTag(obj, objName, matList, matNames, proj) # call function to place tag - UVW
elif objUVW and not objTEX in tags:
# if there is no Tex Tag but UVW tag place tag - UVW
placeTag(obj, objName, matList, matNames, proj)
elif not objUVW and not objTEX in tags:
def placeTag(obj, objName, matList, matNames, proj):
texTag = c4d.TextureTag()
# setMAT gets the material from the in matList, which has an object name matching the material name
setMAT = matList[matNames.index(objName)]
# print matNames.index(objName)
# print "setMAT is: ", setMAT
if proj == 'cubic':
texTag.SetMaterial(setMAT)
texTag[c4d.TEXTURETAG_PROJECTION] = c4d.TEXTURETAG_PROJECTION_CUBIC
obj.InsertTag(texTag)
doc.AddUndo(c4d.UNDOTYPE_NEW, texTag)
# move existing material tags to last tag
# moveExistingTags(obj)
elif proj == 'uvw':
texTag[c4d.TEXTURETAG_PROJECTION] = c4d.TEXTURETAG_PROJECTION_UVW
def moveExistingTags(obj):
''' this is the last function to be executed
and its the function that limits the amount of tags to 2'''
firstTag = obj.GetFirstTag()
lastTag = firstTag
# tagList = []
if lastTag:
while True:
# tagList.append(lastTag)
if not lastTag.GetNext():
lastTag = lastTag.GetNext()
# add a CHANGE undo
doc.AddUndo(c4d.UNDOTYPE_CHANGE, firstTag)
# remove the first tag and insert after last tag
firstTag.Remove()
# here is the LIMITATION of two tags on the object
# we need to insert the firstTag + tagList in reverse order
obj.InsertTag(firstTag, lastTag)
# not sure what we need this list for yet.
# return tagList
# MAIN #############################
def main():
doc = c4d.documents.GetActiveDocument()
obj = doc.GetFirstObject()
# start UNDO
doc.StartUndo()
# call the recursive function with callback
recurseObjs(obj, checkName)
# end UNDO
doc.EndUndo()
# update the Cinema 4D UI
c4d.EventAdd()
if __name__ == '__main__':
main()
"""
Name-US: df_pyApplyTex_v0.1(v15) 12-03-2017
Description-US: Places textures on all objects in hierarchy baed on object->material matching names.
Author: David Flamholc, vfxvoodoo.com
"""
# pyApplyTex_version
pyApply = "df_pyApplyTex_v0.1(v15) 12-03-2017"
"""
LICENSE
-------
Copyright (c) 2016, David Flamholc, VFXVoodoo.com
Programming: David Flamholc <dflamholc@gmail.com>
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
CHANGE LOG
----------
version: paApplyTex_v08
- implemented UNDO functions
version: paApplyTex_v09
- implemented moving newly created tags to last in the stack
version: paApplyTex_v10 -- (df_pyApplyTex_v0.1(v10) 12-03-2017)
- refactoring of funcitons and variables
- and FLOW comments
- scrip header INFO and TO DO list added
# THE FLOW ######
# in the MAIN function we call the recurseObjs function
# recurseObjs iterates through the hierarchy and calls the checkName function on every object
# checkName checks the objects name against the names in the material list and runs placeTags on matchin objects
# gets the material matching the object name from the list of materials and sets the tags projection type,
# then sets the material in the empty textag that was created, sets teh projection
KNOWN BUGS
------------------------
- object that already have a non matching material on them will keep stacking new matching materials endlessly
-
INSTRUCTIONS & DEV NOTES
------------------------
- objects and materials need to have exactly the same name for this script to work
- This Software should work on all os platforms.
- It has only been tested on Cinema 4D r18.
TO DO -- FEATURE IDEAS
-----------------------
[x] Write script comments
[] Create Icon
[] Write Usage Instructions
[] Convert to Plugin
[] Get Unique ID
[] check PART OF NAME so that matching can be done by _TOKEN only on BOTH ON Material or Object
[] fix material stacking BUG on object that already have a non matching tex tag applied (NON MATCHING?? how)
[] ADD feature to control mapping of Tex tag X & LENGTH from text file
Length U == Texture[c4d.TEXTURETAG.LENGTHX] & Length V == Texture[c4d.TEXTURETAG.LENGTHY]
these are presented in 0%-100% but work in normalised float values 0-1
[] cover for None Name TEX Tags
[] if the object has only one non-matching tag then new correct tag is added,
if teh object has two non matching tags then no new tag is added - LIMIT OF TWO MATERIALS
--->> BUT unable to reproduce the error of infinite stacking of tags
"""
# ====== MODULE IMPORT ======#
import c4d
import os
# FUNCTIONS ##############################################################################################
def recurseObjs(obj, checkName):
while obj:
checkName(obj)
recurseObjs(obj.GetDown(), checkName)
obj = obj.GetNext()
return obj
def checkName(obj):
objName = obj.GetName()
matList = doc.GetMaterials()
matNames = [mat.GetName() for mat in matList]
# check for TOKEN in both MATERIAL and OBJECT NAME
if objName in matNames:
checkTags(obj, objName, matList, matNames)
else:
print "No matching names or name _tokens!"
def checkTags(obj, objName, matList, matNames):
objTEX = obj.GetTag(c4d.Ttexture)
objUVW = obj.GetTag(c4d.Tuvw)
tag = obj.GetFirstTag()
tags = obj.GetTags()
proj = None
# check if there is a TEX tag on the object but no UVW tag
if objTEX and not objUVW in tags:
tag = obj.GetFirstTag()
while tag:
if tag == objTEX:
checkTag.append(tag)
if not tag.GetNext():
break
tag = tag.GetNext()
# check the new temp list for name of tex tags
for tTag in checkTag:
if tTag.GetMaterial().GetName() == objName:
return False # do nothing
else:
proj = 'cubic'
placeTag(obj, objName, matList, matNames, proj) # call tag placement funcitons with CUBIC
elif objTEX and objUVW in tags:
# checkTag collects the name of each material tag only
checkTag = [t.GetMaterial().GetName() for i, t in enumerate(tags) if t.GetType() == 5616]
# checking if matching tag is in checkTag and if a matching tag is at the end
if objName in checkTag and objName == checkTag[-1]:
print "we need to move all matching tags that are not at the end, to the end"
print checkTag[:-1]
matchingTags = [t for t in checkTag[:-1] if t == objName]
# the amount of iterations == len(matchingTags)
# if a matching tag were to be in the middle we need to move it second to last.
for i, t in enumerate(checkTag[:-1]):
# the tags order will change as we move the first one
# can we control it with a count ?
count = 0
if t == objName and count >= 0:
print i
print "this is the tag to move to position [-2]"
count += 1
elif count < 0:
print "this tag has to move to position [-3]"
# remove tags - but can't do this based on name
# maybe above instead of creating a list of names instead remove tags based on name
# and then reinsert them
# also check for selection tags
elif objName in checkTag and objName != checkTag[-1]:
print "we'll move the tag to the end"
# tagPos = [m for m in checkTag if objName in checkTag]
moveTag = checkTag.index(objName)
# we need to find the index of the matchin tag in the list tags
pass
# move tag to last tag
# get position of tag
else:
# check what projection in checkTag[-1]
proj = 'uvw'
# if statement to see if the objname appears now
placeTag(obj, objName, matList, matNames, proj) # call function to place tag - UVW
elif objUVW and not objTEX in tags:
proj = 'uvw'
# if there is no Tex Tag but UVW tag place tag - UVW
placeTag(obj, objName, matList, matNames, proj)
elif not objUVW and not objTEX in tags:
proj = 'cubic'
placeTag(obj, objName, matList, matNames, proj)
def placeTag(obj, objName, matList, matNames, proj):
texTag = c4d.TextureTag()
# setMAT gets the material from the in matList, which has an object name matching the material name
setMAT = matList[matNames.index(objName)]
# print matNames.index(objName)
# print "setMAT is: ", setMAT
if proj == 'cubic':
texTag.SetMaterial(setMAT)
texTag[c4d.TEXTURETAG_PROJECTION] = c4d.TEXTURETAG_PROJECTION_CUBIC
obj.InsertTag(texTag)
doc.AddUndo(c4d.UNDOTYPE_NEW, texTag)
# move existing material tags to last tag
# moveExistingTags(obj)
elif proj == 'uvw':
texTag.SetMaterial(setMAT)
texTag[c4d.TEXTURETAG_PROJECTION] = c4d.TEXTURETAG_PROJECTION_UVW
obj.InsertTag(texTag)
doc.AddUndo(c4d.UNDOTYPE_NEW, texTag)
# move existing material tags to last tag
# moveExistingTags(obj)
def moveExistingTags(obj):
''' this is the last function to be executed
and its the function that limits the amount of tags to 2'''
firstTag = obj.GetFirstTag()
lastTag = firstTag
# tagList = []
if lastTag:
while True:
# tagList.append(lastTag)
if not lastTag.GetNext():
break
lastTag = lastTag.GetNext()
# add a CHANGE undo
doc.AddUndo(c4d.UNDOTYPE_CHANGE, firstTag)
# remove the first tag and insert after last tag
firstTag.Remove()
# here is the LIMITATION of two tags on the object
# we need to insert the firstTag + tagList in reverse order
obj.InsertTag(firstTag, lastTag)
# not sure what we need this list for yet.
# return tagList
# MAIN #############################
def main():
doc = c4d.documents.GetActiveDocument()
obj = doc.GetFirstObject()
# start UNDO
doc.StartUndo()
# call the recursive function with callback
recurseObjs(obj, checkName)
# end UNDO
doc.EndUndo()
# update the Cinema 4D UI
c4d.EventAdd()
if __name__ == '__main__':
main()