generated from ynput/ayon-addon-template
-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathlib.py
1598 lines (1251 loc) · 49.5 KB
/
lib.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
# -*- coding: utf-8 -*-
import sys
import os
import errno
import re
import logging
import json
from contextlib import contextmanager
import six
import ayon_api
import hou
from ayon_core.lib import StringTemplate
from ayon_core.settings import get_current_project_settings
from ayon_core.pipeline import (
Anatomy,
registered_host,
get_current_context,
get_current_host_name,
)
from ayon_core.pipeline.create import CreateContext
from ayon_core.pipeline.template_data import get_template_data
from ayon_core.pipeline.context_tools import get_current_task_entity
from ayon_core.pipeline.workfile.workfile_template_builder import (
TemplateProfileNotFound
)
from ayon_core.tools.utils import PopupUpdateKeys, SimplePopup
from ayon_core.tools.utils.host_tools import get_tool_by_name
self = sys.modules[__name__]
self._parent = None
log = logging.getLogger(__name__)
JSON_PREFIX = "JSON:::"
def get_entity_fps(entity=None):
"""Return current task fps or fps from an entity."""
if entity is None:
entity = get_current_task_entity(fields=["attrib.fps"])
return entity["attrib"]["fps"]
def get_output_parameter(node):
"""Return the render output parameter of the given node
Example:
root = hou.node("/obj")
my_alembic_node = root.createNode("alembic")
get_output_parameter(my_alembic_node)
>>> "filename"
Notes:
I'm using node.type().name() to get on par with the creators,
Because the return value of `node.type().name()` is the
same string value used in creators
e.g. instance_data.update({"node_type": "alembic"})
Rop nodes in different network categories have
the same output parameter.
So, I took that into consideration as a hint for
future development.
Args:
node(hou.Node): node instance
Returns:
hou.Parm
"""
node_type = node.type().name()
# Figure out which type of node is being rendered
if node_type in {"alembic", "rop_alembic"}:
return node.parm("filename")
elif node_type == "arnold":
if node_type.evalParm("ar_ass_export_enable"):
return node.parm("ar_ass_file")
return node.parm("ar_picture")
elif node_type in {
"geometry",
"rop_geometry",
"filmboxfbx",
"rop_fbx"
}:
return node.parm("sopoutput")
elif node_type == "comp":
return node.parm("copoutput")
elif node_type in {"karma", "opengl", "flipbook"}:
return node.parm("picture")
elif node_type == "ifd": # Mantra
if node.evalParm("soho_outputmode"):
return node.parm("soho_diskfile")
return node.parm("vm_picture")
elif node_type == "Redshift_Proxy_Output":
return node.parm("RS_archive_file")
elif node_type == "Redshift_ROP":
return node.parm("RS_outputFileNamePrefix")
elif node_type in {"usd", "usd_rop", "usdexport"}:
return node.parm("lopoutput")
elif node_type in {"usdrender", "usdrender_rop"}:
return node.parm("outputimage")
elif node_type == "vray_renderer":
return node.parm("SettingsOutput_img_file_path")
raise TypeError("Node type '%s' not supported" % node_type)
def get_lops_rop_context_options(
ropnode: hou.RopNode) -> "dict[str, str | float]":
"""Return the Context Options that a LOP ROP node uses."""
rop_context_options: "dict[str, str | float]" = {}
# Always set @ropname and @roppath
# See: https://www.sidefx.com/docs/houdini/hom/hou/isAutoContextOption.html
rop_context_options["ropname"] = ropnode.name()
rop_context_options["roppath"] = ropnode.path()
# Set @ropcook, @ropstart, @ropend and @ropinc if setropcook is enabled
setropcook_parm = ropnode.parm("setropcook")
if setropcook_parm:
setropcook = setropcook_parm.eval()
if setropcook:
# TODO: Support "Render Frame Range from Stage" correctly
# TODO: Support passing in the start, end, and increment values
# for the cases where this may need to consider overridden
# frame ranges for `RopNode.render()` calls.
trange = ropnode.evalParm("trange")
if trange == 0:
# Current frame
start: float = hou.frame()
end: float = start
inc: float = 1.0
elif trange in {1, 2}:
# Frame range
start: float = ropnode.evalParm("f1")
end: float = ropnode.evalParm("f2")
inc: float = ropnode.evalParm("f3")
else:
raise ValueError("Unsupported trange value: %s" % trange)
rop_context_options["ropcook"] = 1.0
rop_context_options["ropstart"] = start
rop_context_options["ropend"] = end
rop_context_options["ropinc"] = inc
# Get explicit context options set on the ROP node.
num = ropnode.evalParm("optioncount")
for i in range(1, num + 1):
# Ignore disabled options
if not ropnode.evalParm(f"optionenable{i}"):
continue
name: str = ropnode.evalParm(f"optionname{i}")
option_type: str = ropnode.evalParm(f"optiontype{i}")
if option_type == "string":
value: str = ropnode.evalParm(f"optionstrvalue{i}")
elif option_type == "float":
value: float = ropnode.evalParm(f"optionfloatvalue{i}")
else:
raise ValueError(f"Unsupported option type: {option_type}")
rop_context_options[name] = value
return rop_context_options
@contextmanager
def context_options(context_options: "dict[str, str | float]"):
"""Context manager to set Solaris Context Options.
The original context options are restored after the context exits.
Arguments:
context_options (dict[str, str | float]):
The Solaris Context Options to set.
Yields:
dict[str, str | float]: The original context options that were changed.
"""
# Get the original context options and their values
original_context_options: "dict[str, str | float]" = {}
for name in hou.contextOptionNames():
original_context_options[name] = hou.contextOption(name)
try:
# Override the context options
for name, value in context_options.items():
hou.setContextOption(name, value)
yield original_context_options
finally:
# Restore original context options that we changed
for name in context_options:
if name in original_context_options:
hou.setContextOption(name, original_context_options[name])
else:
# Clear context option
hou.setContextOption(name, None)
@contextmanager
def update_mode_context(mode):
original = hou.updateModeSetting()
try:
hou.setUpdateMode(mode)
yield
finally:
hou.setUpdateMode(original)
def set_scene_fps(fps):
hou.setFps(fps)
# Valid FPS
def validate_fps():
"""Validate current scene FPS and show pop-up when it is incorrect
Returns:
bool
"""
fps = get_entity_fps()
current_fps = hou.fps() # returns float
if current_fps != fps:
# Find main window
parent = hou.ui.mainQtWindow()
if parent is None:
pass
else:
dialog = PopupUpdateKeys(parent=parent)
dialog.setModal(True)
dialog.setWindowTitle("Houdini scene does not match project FPS")
dialog.set_message("Scene %i FPS does not match project %i FPS" %
(current_fps, fps))
dialog.set_button_text("Fix")
# on_show is the Fix button clicked callback
dialog.on_clicked_state.connect(lambda: set_scene_fps(fps))
dialog.show()
return False
return True
def render_rop(ropnode, frame_range=None):
"""Render ROP node utility for Publishing.
This renders a ROP node with the settings we want during Publishing.
Args:
ropnode (hou.RopNode): Node to render
frame_range (tuple): Copied from Houdini's help..
Sequence of 2 or 3 values, overrides the frame range and frame
increment to render. The first two values specify the start and
end frames, and the third value (if given) specifies the frame
increment. If no frame increment is given and the ROP node
doesn't specify a frame increment, then a value of 1 will be
used. If no frame range is given, and the ROP node doesn't
specify a frame range, then the current frame will be rendered.
"""
if frame_range is None:
frame_range = ()
# Print verbose when in batch mode without UI
verbose = not hou.isUIAvailable()
# Render
try:
ropnode.render(verbose=verbose,
# Allow Deadline to capture completion percentage
output_progress=verbose,
# Render only this node
# (do not render any of its dependencies)
ignore_inputs=True,
frame_range=frame_range)
except hou.Error as exc:
# The hou.Error is not inherited from a Python Exception class,
# so we explicitly capture the houdini error, otherwise pyblish
# will remain hanging.
import traceback
traceback.print_exc()
raise RuntimeError("Render failed: {0}".format(exc))
def imprint(node, data, update=False):
"""Store attributes with value on a node
Depending on the type of attribute it creates the correct parameter
template. Houdini uses a template per type, see the docs for more
information.
http://www.sidefx.com/docs/houdini/hom/hou/ParmTemplate.html
Because of some update glitch where you cannot overwrite existing
ParmTemplates on node using:
`setParmTemplates()` and `parmTuplesInFolder()`
update is done in another pass.
Args:
node(hou.Node): node object from Houdini
data(dict): collection of attributes and their value
update (bool, optional): flag if imprint should update
already existing data or leave them untouched and only
add new.
Returns:
None
"""
if not data:
return
if not node:
self.log.error("Node is not set, calling imprint on invalid data.")
return
current_parms = {p.name(): p for p in node.spareParms()}
update_parm_templates = []
new_parm_templates = []
for key, value in data.items():
if value is None:
continue
parm_template = get_template_from_value(key, value)
if key in current_parms:
if node.evalParm(key) == value:
continue
if not update:
log.debug(f"{key} already exists on {node}")
else:
log.debug(f"replacing {key}")
update_parm_templates.append(parm_template)
continue
new_parm_templates.append(parm_template)
if not new_parm_templates and not update_parm_templates:
return
parm_group = node.parmTemplateGroup()
# Add new parm templates
if new_parm_templates:
parm_folder = parm_group.findFolder("Extra")
# if folder doesn't exist yet, create one and append to it,
# else append to existing one
if not parm_folder:
parm_folder = hou.FolderParmTemplate("folder", "Extra")
parm_folder.setParmTemplates(new_parm_templates)
parm_group.append(parm_folder)
else:
# Add to parm template folder instance then replace with updated
# one in parm template group
for template in new_parm_templates:
parm_folder.addParmTemplate(template)
parm_group.replace(parm_folder.name(), parm_folder)
# Update existing parm templates
for parm_template in update_parm_templates:
parm_group.replace(parm_template.name(), parm_template)
# When replacing a parm with a parm of the same name it preserves its
# value if before the replacement the parm was not at the default,
# because it has a value override set. Since we're trying to update the
# parm by using the new value as `default` we enforce the parm is at
# default state
node.parm(parm_template.name()).revertToDefaults()
node.setParmTemplateGroup(parm_group)
def lsattr(attr, value=None, root="/"):
"""Return nodes that have `attr`
When `value` is not None it will only return nodes matching that value
for the given attribute.
Args:
attr (str): Name of the attribute (hou.Parm)
value (object, Optional): The value to compare the attribute too.
When the default None is provided the value check is skipped.
root (str): The root path in Houdini to search in.
Returns:
list: Matching nodes that have attribute with value.
"""
if value is None:
# Use allSubChildren() as allNodes() errors on nodes without
# permission to enter without a means to continue of querying
# the rest
nodes = hou.node(root).allSubChildren()
return [n for n in nodes if n.parm(attr)]
return lsattrs({attr: value})
def lsattrs(attrs, root="/"):
"""Return nodes matching `key` and `value`
Arguments:
attrs (dict): collection of attribute: value
root (str): The root path in Houdini to search in.
Example:
>> lsattrs({"id": "myId"})
["myNode"]
>> lsattr("id")
["myNode", "myOtherNode"]
Returns:
list: Matching nodes that have attribute with value.
"""
matches = set()
# Use allSubChildren() as allNodes() errors on nodes without
# permission to enter without a means to continue of querying
# the rest
nodes = hou.node(root).allSubChildren()
for node in nodes:
for attr in attrs:
if not node.parm(attr):
continue
elif node.evalParm(attr) != attrs[attr]:
continue
else:
matches.add(node)
return list(matches)
def read(node):
"""Read the container data in to a dict
Args:
node(hou.Node): Houdini node
Returns:
dict
"""
# `spareParms` returns a tuple of hou.Parm objects
data = {}
if not node:
return data
for parameter in node.spareParms():
value = parameter.eval()
# test if value is json encoded dict
if isinstance(value, six.string_types) and \
value.startswith(JSON_PREFIX):
try:
value = json.loads(value[len(JSON_PREFIX):])
except json.JSONDecodeError:
# not a json
pass
data[parameter.name()] = value
return data
@contextmanager
def maintained_selection():
"""Maintain selection during context
Example:
>>> with maintained_selection():
... # Modify selection
... node.setSelected(on=False, clear_all_selected=True)
>>> # Selection restored
"""
previous_selection = hou.selectedNodes()
try:
yield
finally:
# Clear the selection
# todo: does hou.clearAllSelected() do the same?
for node in hou.selectedNodes():
node.setSelected(on=False)
if previous_selection:
for node in previous_selection:
node.setSelected(on=True)
@contextmanager
def parm_values(overrides):
"""Override Parameter values during the context.
Arguments:
overrides (List[Tuple[hou.Parm, Any]]): The overrides per parm
that should be applied during context.
"""
originals = []
try:
for parm, value in overrides:
originals.append((parm, parm.eval()))
parm.set(value)
yield
finally:
for parm, value in originals:
# Parameter might not exist anymore so first
# check whether it's still valid
if hou.parm(parm.path()):
parm.set(value)
def reset_framerange(fps=True, frame_range=True):
"""Set frame range and FPS to current folder."""
task_entity = get_current_task_entity(fields={"attrib"})
# Set FPS
if fps:
fps = get_entity_fps(task_entity)
print("Setting scene FPS to {}".format(int(fps)))
set_scene_fps(fps)
if frame_range:
# Set Start and End Frames
task_attrib = task_entity["attrib"]
frame_start = task_attrib.get("frameStart", 0)
frame_end = task_attrib.get("frameEnd", 0)
handle_start = task_attrib.get("handleStart", 0)
handle_end = task_attrib.get("handleEnd", 0)
frame_start -= int(handle_start)
frame_end += int(handle_end)
# Set frame range and FPS
hou.playbar.setFrameRange(frame_start, frame_end)
hou.playbar.setPlaybackRange(frame_start, frame_end)
hou.setFrame(frame_start)
def get_main_window():
"""Acquire Houdini's main window"""
if self._parent is None:
self._parent = hou.ui.mainQtWindow()
return self._parent
def get_template_from_value(key, value):
if isinstance(value, float):
parm = hou.FloatParmTemplate(name=key,
label=key,
num_components=1,
default_value=(value,))
elif isinstance(value, bool):
parm = hou.ToggleParmTemplate(name=key,
label=key,
default_value=value)
elif isinstance(value, int):
parm = hou.IntParmTemplate(name=key,
label=key,
num_components=1,
default_value=(value,))
elif isinstance(value, six.string_types):
parm = hou.StringParmTemplate(name=key,
label=key,
num_components=1,
default_value=(value,))
elif isinstance(value, (dict, list, tuple)):
parm = hou.StringParmTemplate(name=key,
label=key,
num_components=1,
default_value=(
JSON_PREFIX + json.dumps(value),))
else:
raise TypeError("Unsupported type: %r" % type(value))
return parm
def get_frame_data(node, log=None):
"""Get the frame data: `frameStartHandle`, `frameEndHandle`
and `byFrameStep`.
This function uses Houdini node's `trange`, `t1, `t2` and `t3`
parameters as the source of truth for the full inclusive frame
range to render, as such these are considered as the frame
range including the handles.
The non-inclusive frame start and frame end without handles
can be computed by subtracting the handles from the inclusive
frame range.
Args:
node (hou.Node): ROP node to retrieve frame range from,
the frame range is assumed to be the frame range
*including* the start and end handles.
Returns:
dict: frame data for `frameStartHandle`, `frameEndHandle`
and `byFrameStep`.
"""
if log is None:
log = self.log
data = {}
if node.parm("trange") is None:
log.debug(
"Node has no 'trange' parameter: {}".format(node.path())
)
return data
if node.evalParm("trange") == 0:
data["frameStartHandle"] = hou.intFrame()
data["frameEndHandle"] = hou.intFrame()
data["byFrameStep"] = 1.0
log.info(
"Node '{}' has 'Render current frame' set.\n"
"Task handles are ignored.\n"
"frameStart and frameEnd are set to the "
"current frame.".format(node.path())
)
else:
data["frameStartHandle"] = int(node.evalParm("f1"))
data["frameEndHandle"] = int(node.evalParm("f2"))
data["byFrameStep"] = node.evalParm("f3")
return data
def splitext(name, allowed_multidot_extensions):
# type: (str, list) -> tuple
"""Split file name to name and extension.
Args:
name (str): File name to split.
allowed_multidot_extensions (list of str): List of allowed multidot
extensions.
Returns:
tuple: Name and extension.
"""
for ext in allowed_multidot_extensions:
if name.endswith(ext):
return name[:-len(ext)], ext
return os.path.splitext(name)
def get_top_referenced_parm(parm):
processed = set() # disallow infinite loop
while True:
if parm.path() in processed:
raise RuntimeError("Parameter references result in cycle.")
processed.add(parm.path())
ref = parm.getReferencedParm()
if ref.path() == parm.path():
# It returns itself when it doesn't reference
# another parameter
return ref
else:
parm = ref
def evalParmNoFrame(node, parm, pad_character="#"):
parameter = node.parm(parm)
assert parameter, "Parameter does not exist: %s.%s" % (node, parm)
# If the parameter has a parameter reference, then get that
# parameter instead as otherwise `unexpandedString()` fails.
parameter = get_top_referenced_parm(parameter)
# Substitute out the frame numbering with padded characters
try:
raw = parameter.unexpandedString()
except hou.Error as exc:
print("Failed: %s" % parameter)
raise RuntimeError(exc)
def replace(match):
padding = 1
n = match.group(2)
if n and int(n):
padding = int(n)
return pad_character * padding
expression = re.sub(r"(\$F([0-9]*))", replace, raw)
with hou.ScriptEvalContext(parameter):
return hou.expandStringAtFrame(expression, 0)
def get_color_management_preferences():
"""Get default OCIO preferences"""
preferences = {
"config": hou.Color.ocio_configPath(),
"display": hou.Color.ocio_defaultDisplay(),
"view": hou.Color.ocio_defaultView()
}
# Note: For whatever reason they are cases where `view` may be an empty
# string even though a valid default display is set where `PyOpenColorIO`
# does correctly return the values.
# Workaround to get the correct default view
if preferences["config"] and not preferences["view"]:
log.debug(
"Houdini `hou.Color.ocio_defaultView()` returned empty value."
" Falling back to `PyOpenColorIO` to get the default view.")
try:
import PyOpenColorIO
except ImportError:
log.warning(
"Unable to workaround empty return value of "
"`hou.Color.ocio_defaultView()` because `PyOpenColorIO` is "
"not available.")
return preferences
config_path = preferences["config"]
config = PyOpenColorIO.Config.CreateFromFile(config_path)
display = config.getDefaultDisplay()
assert display == preferences["display"], \
"Houdini default OCIO display must match config default display"
view = config.getDefaultView(display)
preferences["display"] = display
preferences["view"] = view
return preferences
def get_obj_node_output(obj_node):
"""Find output node.
If the node has any output node return the
output node with the minimum `outputidx`.
When no output is present return the node
with the display flag set. If no output node is
detected then None is returned.
Arguments:
node (hou.Node): The node to retrieve a single
the output node for.
Returns:
Optional[hou.Node]: The child output node.
"""
outputs = obj_node.subnetOutputs()
if not outputs:
return
elif len(outputs) == 1:
return outputs[0]
else:
return min(outputs,
key=lambda node: node.evalParm('outputidx'))
def get_output_children(output_node, include_sops=True):
"""Recursively return a list of all output nodes
contained in this node including this node.
It works in a similar manner to output_node.allNodes().
"""
out_list = [output_node]
if output_node.childTypeCategory() == hou.objNodeTypeCategory():
for child in output_node.children():
out_list += get_output_children(child, include_sops=include_sops)
elif include_sops and \
output_node.childTypeCategory() == hou.sopNodeTypeCategory():
out = get_obj_node_output(output_node)
if out:
out_list += [out]
return out_list
def get_resolution_from_entity(entity):
"""Get resolution from the given entity.
Args:
entity (dict[str, Any]): Project, Folder or Task entity.
Returns:
Union[Tuple[int, int], None]: Resolution width and height.
"""
if not entity or "attrib" not in entity:
raise ValueError(f"Entity is not valid: \"{entity}\"")
attributes = entity["attrib"]
resolution_width = attributes.get("resolutionWidth")
resolution_height = attributes.get("resolutionHeight")
# Make sure both width and height are set
if resolution_width is None or resolution_height is None:
print(f"No resolution information found in entity: '{entity}'")
return None
return int(resolution_width), int(resolution_height)
def set_camera_resolution(camera, entity=None):
"""Apply resolution to camera from task or folder entity.
Arguments:
camera (hou.OpNode): Camera node.
entity (Optional[Dict[str, Any]]): Folder or task entity.
If not provided falls back to current task entity.
"""
if not entity:
entity = get_current_task_entity()
resolution = get_resolution_from_entity(entity)
if resolution:
print("Setting camera resolution: {} -> {}x{}".format(
camera.name(), resolution[0], resolution[1]
))
camera.parm("resx").set(resolution[0])
camera.parm("resy").set(resolution[1])
def get_camera_from_container(container):
"""Get camera from container node. """
cameras = container.recursiveGlob(
"*",
filter=hou.nodeTypeFilter.ObjCamera,
include_subnets=False
)
assert len(cameras) == 1, "Camera instance must have only one camera"
return cameras[0]
def get_current_context_template_data_with_entity_attrs():
"""Return template data including current context folder and task attribs.
Output contains:
- Regular template data from `get_template_data`
- 'folderAttributes' key with folder attribute values.
- 'taskAttributes' key with task attribute values.
Returns:
dict[str, Any]: Template data to fill templates.
"""
context = get_current_context()
project_name = context["project_name"]
folder_path = context["folder_path"]
task_name = context["task_name"]
host_name = get_current_host_name()
project_entity = ayon_api.get_project(project_name)
anatomy = Anatomy(project_name, project_entity=project_entity)
folder_entity = ayon_api.get_folder_by_path(project_name, folder_path)
task_entity = ayon_api.get_task_by_name(
project_name, folder_entity["id"], task_name
)
# get context specific vars
folder_attributes = folder_entity["attrib"]
task_attributes = task_entity["attrib"]
# compute `frameStartHandle` and `frameEndHandle`
for attributes in [folder_attributes, task_attributes]:
frame_start = attributes.get("frameStart")
frame_end = attributes.get("frameEnd")
handle_start = attributes.get("handleStart")
handle_end = attributes.get("handleEnd")
if frame_start is not None and handle_start is not None:
attributes["frameStartHandle"] = frame_start - handle_start
if frame_end is not None and handle_end is not None:
attributes["frameEndHandle"] = frame_end + handle_end
template_data = get_template_data(
project_entity, folder_entity, task_entity, host_name
)
template_data["root"] = anatomy.roots
template_data["folderAttributes"] = folder_attributes
template_data["taskAttributes"] = task_attributes
return template_data
def set_review_color_space(node, review_color_space="", log=None):
"""Set ociocolorspace parameter for the given OpenGL node.
Set `ociocolorspace` parameter of the given node
to to the given review_color_space value.
If review_color_space is empty, a default colorspace corresponding to
the display & view of the current Houdini session will be used.
Note:
This function expects nodes of type `opengl` or `flipbook`.
Args:
node (hou.Node): ROP node to set its ociocolorspace parm.
review_color_space (str): Colorspace value for ociocolorspace parm.
log (logging.Logger): Logger to log to.
"""
if log is None:
log = self.log
if node.type().name() not in {"opengl", "flipbook"}:
log.warning(
"Type of given node {} not allowed."
" only types `opengl` and `flipbook` are allowed."
.format(node.type().name())
)
# Set Color Correction parameter to OpenColorIO
colorcorrect_parm = node.parm("colorcorrect")
if colorcorrect_parm.evalAsString() != "ocio":
idx = colorcorrect_parm.menuItems().index("ocio")
colorcorrect_parm.set(idx)
log.debug(
"'Color Correction' parm on '{}' has been set to '{}'"
.format(node.path(), colorcorrect_parm.menuLabels()[idx])
)
node.setParms(
{"ociocolorspace": review_color_space}
)
log.debug(
"'OCIO Colorspace' parm on '{}' has been set to "
"the view color space '{}'"
.format(node.path(), review_color_space)
)
def get_context_var_changes():
"""get context var changes."""
houdini_vars_to_update = {}
project_settings = get_current_project_settings()
houdini_vars_settings = \
project_settings["houdini"]["general"]["update_houdini_var_context"]
if not houdini_vars_settings["enabled"]:
return houdini_vars_to_update
houdini_vars = houdini_vars_settings["houdini_vars"]
# No vars specified - nothing to do
if not houdini_vars:
return houdini_vars_to_update
# Get Template data
template_data = get_current_context_template_data_with_entity_attrs()
# Set Houdini Vars
for item in houdini_vars:
# For consistency reasons we always force all vars to be uppercase
# Also remove any leading, and trailing whitespaces.
var = item["var"].strip().upper()
# get and resolve template in value
item_value = StringTemplate.format_template(
item["value"],
template_data
)
if var == "JOB" and item_value == "":
# sync $JOB to $HIP if $JOB is empty
item_value = os.environ["HIP"]
if item["is_directory"]:
item_value = item_value.replace("\\", "/")
current_value = hou.hscript("echo -n `${}`".format(var))[0]
if current_value != item_value:
houdini_vars_to_update[var] = (
current_value, item_value, item["is_directory"]
)
return houdini_vars_to_update
def update_houdini_vars_context():
"""Update task context variables"""
for var, (_old, new, is_directory) in get_context_var_changes().items():