forked from mattebb/3delightblender
-
Notifications
You must be signed in to change notification settings - Fork 133
/
Copy pathrman_scene_sync.py
1285 lines (1129 loc) · 65.5 KB
/
rman_scene_sync.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
# utils
from .rfb_utils import object_utils
from .rfb_utils import texture_utils
from .rfb_utils import scene_utils
from .rfb_utils import shadergraph_utils
from .rfb_utils.timer_utils import time_this
from .rfb_logger import rfb_log
from .rman_sg_nodes.rman_sg_lightfilter import RmanSgLightFilter
from . import rman_constants
from copy import deepcopy
import bpy
class RmanUpdate:
'''
The RmanUpdate class. A helper class to indicate what kind of update
we're dealing with
Attributes:
is_updated_geometry (bool) - Whether the geometry has been updated. This doesn't necessarily
mean the actual geometry was updated. Attribute changes are also
considered geometry updates
is_updated_transform (bool) - Whether the geometry was transformed
is_updated_shading (bool) - If the shader on the object has changed
is_updated_attributes (bool) - Wether an Ri attribute was changed
updated_prop_name (str) - The name of the Blender property that was changed, for either
is_updated_attributes or is_updated_geometry case
do_clear_instances (bool) - Whether we should clear/delete all instances of the prototype
'''
def __init__(self):
self.is_updated_geometry = False
self.is_updated_transform = False
self.is_updated_shading = False
self.is_updated_attributes = False
self.updated_prop_name = None
self.do_clear_instances = True
class RmanSceneSync(object):
'''
The RmanSceneSync class handles keeping the RmanScene object in sync
during IPR.
Attributes:
rman_render (RmanRender) - pointer back to the current RmanRender object
rman () - rman python module
rman_scene (RmanScene) - pointer to the current RmanScene object
sg_scene (RixSGSCene) - the RenderMan scene graph object
'''
def __init__(self, rman_render=None, rman_scene=None, sg_scene=None):
self.rman_render = rman_render
self.rman = rman_render.rman
self.rman_scene = rman_scene
self.sg_scene = sg_scene
self.num_instances_changed = False # if the number of instances has changed since the last update
self.frame_number_changed = False
self.check_all_instances = False # force checking all instances
self.rman_updates = dict() # A dicitonary to hold RmanUpdate instances
self.selected_channel = None
@property
def sg_scene(self):
return self.__sg_scene
@sg_scene.setter
def sg_scene(self, sg_scene):
self.__sg_scene = sg_scene
def reset(self):
self.num_instances_changed = False
self.frame_number_changed = False
self.check_all_instances = False
self.rman_updates = dict()
self.selected_channel = None
def update_view(self, context, depsgraph):
camera = depsgraph.scene.camera
self.rman_scene.context = context
self.rman_scene.depsgraph = depsgraph
self.rman_scene.bl_scene = depsgraph.scene_eval
self.rman_scene.bl_view_layer = depsgraph.view_layer_eval
rman_sg_camera = self.rman_scene.main_camera
translator = self.rman_scene.rman_translators['CAMERA']
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
if self.rman_scene.is_viewport_render:
if translator.update_viewport_resolution(rman_sg_camera):
translator.update_viewport_cam(None, rman_sg_camera, force_update=True)
translator.update_transform(None, rman_sg_camera)
else:
translator.update_transform(camera, rman_sg_camera)
def create_rman_update(self, ob_key, **kwargs):
rman_update = RmanUpdate()
rman_update.is_updated_shading = kwargs.get('update_shading', False)
rman_update.is_updated_transform = kwargs.get('update_transform', False)
rman_update.is_updated_geometry = kwargs.get('update_geometry', False)
rman_update.is_updated_attributes = kwargs.get('update_attributes', False)
rman_update.updated_prop_name = kwargs.get('prop_name', None)
rman_update.do_clear_instances = kwargs.get('clear_instances', True)
self.rman_updates[ob_key] = rman_update
return rman_update
@time_this
def scene_updated(self):
# Check visible objects
visible_objects = self.rman_scene.context.visible_objects
if not self.num_instances_changed:
if len(visible_objects) != self.rman_scene.num_objects_in_viewlayer:
rfb_log().debug("\tNumber of visible objects changed: %d -> %d" % (self.rman_scene.num_objects_in_viewlayer, len(visible_objects)))
# The number of visible objects has changed.
# Figure out the difference using sets
set1 = set(self.rman_scene.objects_in_viewlayer)
set2 = set(visible_objects)
set_diff1 = set1.difference(set2)
set_diff2 = set2.difference(set1)
objects = list(set_diff1.union(set_diff2))
for o in list(objects):
try:
if o.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_shading = True
rman_update.is_updated_transform = True
self.rman_updates[o.original] = rman_update
except:
continue
#self.check_all_instances = True
self.rman_scene.num_objects_in_viewlayer = len(visible_objects)
self.rman_scene.objects_in_viewlayer = [o for o in visible_objects]
if self.rman_scene.bl_frame_current != self.rman_scene.bl_scene.frame_current:
# frame changed, update any materials and objects that
# are marked as frame sensitive
rfb_log().debug("Frame changed: %d -> %d" % (self.rman_scene.bl_frame_current, self.rman_scene.bl_scene.frame_current))
self.rman_scene.bl_frame_current = self.rman_scene.bl_scene.frame_current
self.frame_number_changed = True
# check for frame sensitive objects
for id in self.rman_scene.depsgraph.ids:
if isinstance(id, bpy.types.Object):
o = id.original
if o.type == 'CAMERA':
rman_sg_node = self.rman_scene.rman_cameras.get(o.original, None)
else:
rman_sg_node = self.rman_scene.get_rman_prototype(object_utils.prototype_key(o), create=False)
if rman_sg_node and rman_sg_node.is_frame_sensitive:
if o.original not in self.rman_updates:
if o.type == 'CAMERA':
o.original.update_tag()
else:
# manually create an RmanUpdate
# calling update_tag() on a EMPTY doesn't seem to trigger
# a updated_geometry update
rman_update = RmanUpdate()
rman_update.is_updated_geometry = True
rman_update.is_updated_transform = False
self.rman_updates[o.original] = rman_update
elif isinstance(id, bpy.types.Material):
mat = id.original
rman_sg_material = self.rman_scene.rman_materials.get(mat, None)
if rman_sg_material and rman_sg_material.is_frame_sensitive:
mat.node_tree.update_tag()
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
# update frame number
options = self.rman_scene.sg_scene.GetOptions()
options.SetInteger(self.rman.Tokens.Rix.k_Ri_Frame, self.rman_scene.bl_frame_current)
self.rman_scene.sg_scene.SetOptions(options)
def _mesh_light_update(self, mat):
object_list = list()
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
for ob_inst in self.rman_scene.depsgraph.object_instances:
ob = ob_inst.object.evaluated_get(self.rman_scene.depsgraph)
if not hasattr(ob.data, 'materials'):
continue
if ob.type in ('ARMATURE', 'CURVE', 'CAMERA'):
continue
proto_key = object_utils.prototype_key(ob_inst)
rman_sg_node = self.rman_scene.get_rman_prototype(proto_key)
if rman_sg_node:
found = False
for name, material in ob.data.materials.items():
if name == mat.name:
found = True
if found:
del self.rman_scene.rman_prototypes[proto_key]
if ob not in object_list:
object_list.append(ob)
for ob in object_list:
ob.update_tag()
def material_updated(self, ob_update, rman_sg_material=None):
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
mat = ob_update.id
else:
mat = ob_update
if rman_sg_material is None:
rman_sg_material = self.rman_scene.rman_materials.get(mat.original, None)
translator = self.rman_scene.rman_translators["MATERIAL"]
db_name = object_utils.get_db_name(mat)
if not rman_sg_material:
# Double check if we can't find the material because of an undo
rman_sg_material = self.update_materials_dict(mat)
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
if not rman_sg_material:
rfb_log().debug("New material: %s" % mat.name)
db_name = object_utils.get_db_name(mat)
rman_sg_material = translator.export(mat, db_name)
self.rman_scene.rman_materials[mat.original] = rman_sg_material
else:
rfb_log().debug("Material, call update")
translator.update(mat, rman_sg_material)
# update db_name
rman_sg_material.db_name = db_name
def light_filter_transform_updated(self, ob, rman_sg_lightfilter):
translator = self.rman_scene.rman_translators['LIGHTFILTER']
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
remove_items = []
for light_ob in rman_sg_lightfilter.lights_list:
try:
is_mesh_light = shadergraph_utils.is_mesh_light(light_ob)
if light_ob.name not in bpy.data.objects:
remove_items.append(light_ob)
continue
child = None
if is_mesh_light:
if light_ob.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_geometry = True
rman_update.is_updated_transform = True
self.rman_updates[light_ob.original] = rman_update
else:
proto_key = object_utils.prototype_key(light_ob)
rman_sg_light = self.rman_scene.get_rman_prototype(proto_key)
# try to look for the lightfilter transform group
for i in range(rman_sg_light.sg_node.GetNumChildren()):
c = rman_sg_light.sg_node.GetChild(i)
if c.GetIdentifier().CStr() == rman_sg_lightfilter.db_name:
child = c
break
if child:
translator.update_transform(ob, light_ob, c)
except ReferenceError:
remove_items.append(light_ob)
continue
for item in remove_items:
rman_sg_lightfilter.lights_list.remove(item)
def light_filter_updated(self, ob_update, force_update=False):
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
ob = ob_update.id.evaluated_get(self.rman_scene.depsgraph)
else:
ob = ob_update.evaluated_get(self.rman_scene.depsgraph)
proto_key = object_utils.prototype_key(ob)
rman_sg_lightfilter = self.rman_scene.get_rman_prototype(proto_key)
if not rman_sg_lightfilter:
# Light filter needs to be added
rman_update = RmanUpdate()
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
rman_update.is_updated_geometry = ob_update.is_updated_geometry
rman_update.is_updated_shading = ob_update.is_updated_shading
rman_update.is_updated_transform = ob_update.is_updated_transform
else:
rman_update.is_updated_geometry = True
rman_update.is_updated_shading = True
rman_update.is_updated_transform = True
self.rman_updates[ob.original] = rman_update
return
if force_update or ob_update.is_updated_transform or ob_update.is_updated_shading:
rfb_log().debug("\tLight Filter: %s Transform Updated" % ob.name)
self.light_filter_transform_updated(ob, rman_sg_lightfilter)
if force_update or ob_update.is_updated_geometry:
rfb_log().debug("\tLight Filter: %s Shading Updated" % ob.name)
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
self.rman_scene.rman_translators['LIGHTFILTER'].update(ob, rman_sg_lightfilter)
remove_items = []
for light_ob in rman_sg_lightfilter.lights_list:
try:
is_mesh_light = shadergraph_utils.is_mesh_light(light_ob)
if light_ob.name not in bpy.data.objects:
remove_items.append(light_ob)
continue
if is_mesh_light:
mat = object_utils.get_active_material(light_ob)
self.material_updated(mat)
elif light_ob.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_geometry = True
rman_update.is_updated_transform = True
self.rman_updates[light_ob.original] = rman_update
except ReferenceError:
remove_items.append(light_ob)
continue
for item in remove_items:
rman_sg_lightfilter.light_lights.remove(item)
def camera_updated(self, ob_update, force_update=False):
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
ob = ob_update.id.evaluated_get(self.rman_scene.depsgraph)
else:
ob = ob_update
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
rman_sg_camera = self.rman_scene.rman_cameras.get(ob.original)
translator = self.rman_scene.rman_translators['CAMERA']
rman_update = RmanUpdate()
self.rman_updates[ob.original] = rman_update
if not rman_sg_camera:
rfb_log().debug("\tNew Camera: %s" % ob.name)
db_name = object_utils.get_db_name(ob)
rman_sg_camera = translator._export_render_cam(ob, db_name)
self.rman_scene.rman_cameras[ob.original] = rman_sg_camera
self.rman_scene.sg_scene.Root().AddChild(rman_sg_camera.sg_node)
self.rman_scene.sg_scene.Root().AddCoordinateSystem(rman_sg_camera.sg_node)
return
if force_update or ob_update.is_updated_geometry:
rfb_log().debug("\tUpdated Camera: %s" % ob.name)
if not self.rman_scene.is_viewport_render:
translator.update(ob, rman_sg_camera)
else:
translator.update_viewport_cam(ob, rman_sg_camera, force_update=True)
if force_update or ob_update.is_updated_transform:
# we deal with main camera transforms in view_draw
if self.rman_scene.is_viewport_render and self.rman_scene.main_camera == rman_sg_camera:
return
if translator._update_render_cam_transform(ob, rman_sg_camera):
rfb_log().debug("\tCamera Transform Updated: %s" % ob.name)
def check_particle_instancer(self, ob_update, psys):
# this particle system is a instancer
inst_ob = getattr(psys.settings, 'instance_object', None)
collection = getattr(psys.settings, 'instance_collection', None)
if inst_ob:
if inst_ob.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_shading = ob_update.is_updated_shading
rman_update.is_updated_transform = ob_update.is_updated_transform
self.rman_updates[inst_ob.original] = rman_update
elif collection:
for col_obj in collection.all_objects:
if not col_obj.original.data:
continue
if col_obj.original in self.rman_updates:
continue
rman_update = RmanUpdate()
rman_update.is_updated_shading = ob_update.is_updated_shading
rman_update.is_updated_transform = ob_update.is_updated_transform
self.rman_updates[col_obj.original] = rman_update
def update_particle_emitter(self, ob, psys, delete=False):
psys_translator = self.rman_scene.rman_translators['PARTICLES']
proto_key = object_utils.prototype_key(ob)
rman_sg_particles = self.rman_scene.get_rman_particles(proto_key, psys, ob, create=not delete)
if rman_sg_particles:
psys_translator.update(ob, psys, rman_sg_particles)
def update_particle_emitters(self, ob):
for psys in ob.particle_systems:
if not object_utils.is_particle_instancer(psys):
self.update_particle_emitter(ob, psys)
def update_empty(self, ob_update, rman_sg_node=None):
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
ob = ob_update.id
else:
ob = ob_update.evaluated_get(self.rman_scene.depsgraph)
rfb_log().debug("Update empty: %s" % ob.name)
if ob.is_instancer:
rfb_log().debug("\tEmpty is an instancer")
collection = ob.instance_collection
if collection:
for col_obj in collection.all_objects:
if not col_obj.original.data:
continue
if col_obj.original in self.rman_updates:
continue
rman_update = RmanUpdate()
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
rman_update.is_updated_geometry = ob_update.is_updated_geometry
rman_update.is_updated_shading = ob_update.is_updated_shading
rman_update.is_updated_transform = ob_update.is_updated_transform
else:
rman_update.is_updated_geometry = True
rman_update.is_updated_shading = True
rman_update.is_updated_transform = True
self.rman_updates[col_obj.original] = rman_update
else:
rfb_log().debug("\tRegular empty")
proto_key = object_utils.prototype_key(ob)
rman_sg_node = self.rman_scene.get_rman_prototype(proto_key)
if not rman_sg_node:
return
translator = self.rman_scene.rman_translators['EMPTY']
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
translator.export_transform(ob, rman_sg_node.sg_node)
translator.export_object_attributes(ob, rman_sg_node)
self.rman_scene.attach_material(ob, rman_sg_node)
if ob.renderman.export_as_coordsys:
self.rman_scene.get_root_sg_node().AddCoordinateSystem(rman_sg_node.sg_node)
else:
self.rman_scene.get_root_sg_node().RemoveCoordinateSystem(rman_sg_node.sg_node)
def update_materials_dict(self, mat):
# Try to see if we already have a material with the same db_name
# We need to do this because undo/redo causes all bpy.types.ID
# references to be invalidated (see: https://docs.blender.org/api/current/info_gotcha.html)
# We don't want to accidentally mistake this for a new object, so we need to update
# our materials dictionary with the new bpy.types.ID reference
rman_sg_material = None
for id, rman_sg_node in self.rman_scene.rman_materials.items():
if rman_sg_node:
db_name = object_utils.get_db_name(mat)
if rman_sg_node.db_name == db_name:
self.rman_scene.rman_materials[mat.original] = rman_sg_node
del self.rman_scene.rman_materials[id]
rman_sg_material = rman_sg_node
break
return rman_sg_material
def update_collection(self, coll):
# mark all objects in a collection
# as needing their instances updated
for o in coll.all_objects:
if o.type in ('CAMERA'):
continue
if o.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_shading = True
rman_update.is_updated_transform = True
self.rman_updates[o.original] = rman_update
def check_empty_instancer(self, dps_update):
ob = dps_update.id.evaluated_get(self.rman_scene.depsgraph)
coll = ob.instance_collection
rfb_log().debug("\tEmpty Instancer %s Updated" % ob.name)
rman_update = self.rman_updates.get(ob.original, None)
if rman_update and rman_update.is_updated_attributes:
# this is just an attribute edit
return
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
proto_key = object_utils.prototype_key(ob)
rman_sg_node = self.rman_scene.get_rman_prototype(proto_key, ob=ob, create=True)
rman_type = object_utils._detect_primitive_(ob)
self.rman_scene.rman_translators[rman_type].clear_children(rman_sg_node)
self.rman_scene.attach_material(ob, rman_sg_node, sg_node=rman_sg_node.sg_attributes)
# mark all objects in the instance collection
# as needing their instances updated
for o in coll.all_objects:
if o.type in ('CAMERA'):
continue
if o.original not in self.rman_updates:
rfb_log().debug("\t %s needs updating." % o.original.name)
child_rman_update = RmanUpdate()
child_rman_update.is_updated_geometry = dps_update.is_updated_geometry
child_rman_update.is_updated_shading = dps_update.is_updated_shading
child_rman_update.is_updated_transform = dps_update.is_updated_transform
self.rman_updates[o.original] = child_rman_update
def update_portals(self, ob):
for portal in scene_utils.get_all_portals(ob):
portal.original.update_tag()
def check_light_datablock(self, ob_update):
if isinstance(ob_update, bpy.types.DepsgraphUpdate):
ob = ob_update.id.original
else:
ob = ob_update.original
users = self.rman_scene.context.blend_data.user_map(subset={ob}, value_types={'OBJECT'})
for o in users[ob]:
rman_type = object_utils._detect_primitive_(o)
if rman_type == 'LIGHTFILTER':
self.light_filter_updated(o, force_update=True)
elif o.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_geometry = True
rman_update.is_updated_shading = True
self.rman_updates[o.original] = rman_update
def check_focus_object(self, ob):
for camera in bpy.data.cameras:
rm = camera.renderman
if rm.rman_focus_object and rm.rman_focus_object.original == ob.original:
users = self.rman_scene.context.blend_data.user_map(subset={camera})
for o in users[camera]:
self.camera_updated(o.original, force_update=True)
def check_object_datablock(self, dps_update):
ob_eval = dps_update.id.evaluated_get(self.rman_scene.depsgraph)
rman_type = object_utils._detect_primitive_(ob_eval)
#if ob_eval.type in ('ARMATURE'):
# return
# These types need special handling
if rman_type == 'EMPTY':
self.update_empty(dps_update)
elif rman_type == 'LIGHTFILTER':
self.light_filter_updated(dps_update)
elif rman_type == 'CAMERA':
self.camera_updated(dps_update)
elif rman_type == 'EMPTY_INSTANCER':
self.check_empty_instancer(dps_update)
else:
if dps_update.id.original not in self.rman_updates:
rfb_log().debug("\tObject: %s Updated" % dps_update.id.name)
rfb_log().debug("\t is_updated_geometry: %s" % str(dps_update.is_updated_geometry))
rfb_log().debug("\t is_updated_shading: %s" % str(dps_update.is_updated_shading))
rfb_log().debug("\t is_updated_transform: %s" % str(dps_update.is_updated_transform))
rman_update = RmanUpdate()
rman_update.is_updated_geometry = dps_update.is_updated_geometry
rman_update.is_updated_shading = dps_update.is_updated_shading
rman_update.is_updated_transform = dps_update.is_updated_transform
if not rman_update.is_updated_geometry:
rman_update.do_clear_instances = False
self.rman_updates[dps_update.id.original] = rman_update
for psys in ob_eval.particle_systems:
if object_utils.is_particle_instancer(psys):
self.check_particle_instancer(dps_update, psys)
# Check if this object is the focus object the camera. If it is
# we need to update the camera
if dps_update.is_updated_transform:
self.check_focus_object(ob_eval)
def check_particle_settings(self, dps_update):
rfb_log().debug("ParticleSettings updated: %s" % dps_update.id.name)
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
users = self.rman_scene.context.blend_data.user_map(subset={dps_update.id.original}, value_types={'OBJECT'})
for o in users[dps_update.id.original]:
ob = o.evaluated_get(self.rman_scene.depsgraph)
psys = None
for ps in ob.particle_systems:
if ps.settings.original == dps_update.id.original:
psys = ps
break
if not psys:
continue
if object_utils.is_particle_instancer(psys):
# if this particle system was changed to an instancer
# make sure any old emitters/hair is removed
self.update_particle_emitter(ob, psys, delete=True)
else:
self.update_particle_emitter(ob, psys)
if o.original not in self.rman_updates:
rman_update = RmanUpdate()
rman_update.is_updated_shading = dps_update.is_updated_shading
rman_update.is_updated_transform = dps_update.is_updated_transform
self.rman_updates[o.original] = rman_update
def update_shader_nodetree(self, id):
users = self.rman_scene.context.blend_data.user_map(subset={id})
for o in users[id]:
if self.rman_scene.is_interactive:
if hasattr(o, 'rman_nodetree'):
o.rman_nodetree.update_tag()
elif isinstance(o, bpy.types.Material):
self.update_material(o)
elif isinstance(o, bpy.types.Light):
self.check_light_datablock(o)
elif isinstance(o, bpy.types.ShaderNodeTree):
# this is another shader node tree
# presumbably it's a group node
# just recurse
self.update_shader_nodetree(o)
elif hasattr(o, 'node_tree'):
o.node_tree.update_tag()
else:
if isinstance(o, bpy.types.Light):
self.check_light_datablock(o)
elif isinstance(o, bpy.types.Material):
rman_update = RmanUpdate()
self.rman_updates[o.original] = rman_update
def check_shader_nodetree(self, dps_update):
if dps_update.id.name in bpy.data.node_groups:
if len(dps_update.id.nodes) < 1:
return
if not dps_update.id.name.startswith(rman_constants.RMAN_FAKE_NODEGROUP):
return
# this is one of our fake node groups with ramps
# update all of the users of this node tree
rfb_log().debug("ShaderNodeTree updated: %s" % dps_update.id.name)
self.update_shader_nodetree(dps_update.id.original)
@time_this
def batch_update_scene(self, context, depsgraph):
self.rman_scene.bl_frame_current = self.rman_scene.bl_scene.frame_current
self.rman_updates = dict()
self.num_instances_changed = False
self.check_all_instances = False
self.rman_scene.bl_scene = depsgraph.scene_eval
self.rman_scene.context = context
self.rman_scene.bl_view_layer = depsgraph.view_layer_eval
# update the frame number
options = self.rman_scene.sg_scene.GetOptions()
options.SetInteger(self.rman.Tokens.Rix.k_Ri_Frame, self.rman_scene.bl_frame_current)
self.rman_scene.sg_scene.SetOptions(options)
# Check the number of instances. If we differ, an object may have been
# added or deleted
if self.rman_scene.num_object_instances != len(depsgraph.object_instances):
rfb_log().debug("\tNumber of instances changed: %d -> %d" % (self.rman_scene.num_object_instances, len(depsgraph.object_instances)))
self.num_instances_changed = True
self.rman_scene.num_object_instances = len(depsgraph.object_instances)
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
for dps_update in reversed(depsgraph.updates):
if isinstance(dps_update.id, bpy.types.ParticleSettings):
self.check_particle_settings(dps_update)
elif isinstance(dps_update.id, bpy.types.Light):
self.check_light_datablock(dps_update)
elif isinstance(dps_update.id, bpy.types.Material):
rfb_log().debug("Material updated: %s" % dps_update.id.name)
ob = dps_update.id
if ob.original not in self.rman_updates:
rman_update = RmanUpdate()
self.rman_updates[ob.original] = rman_update
elif isinstance(dps_update.id, bpy.types.ShaderNodeTree):
self.check_shader_nodetree(dps_update)
elif isinstance(dps_update.id, bpy.types.Object):
self.check_object_datablock(dps_update)
elif isinstance(dps_update.id, bpy.types.GeometryNodeTree):
# create an empty RmanUpdate
self.create_rman_update(dps_update.id.original, clear_instances=False)
if not self.rman_updates and self.num_instances_changed:
# The number of instances changed, but we are not able
# to determine what changed. We are forced to check
# all instances.
rfb_log().debug("Set check_all_instances to True")
self.check_all_instances = True
self.check_instances(batch_mode=True)
# update any materials
for id, rman_sg_material in self.rman_scene.rman_materials.items():
if rman_sg_material.is_frame_sensitive or id.original in self.rman_updates:
mat = id.evaluated_get(self.rman_scene.depsgraph)
self.material_updated(mat, rman_sg_material)
self.rman_scene.export_integrator()
self.rman_scene.export_samplefilters()
self.rman_scene.export_displayfilters()
self.rman_scene.export_displays()
if self.rman_scene.do_motion_blur and self.rman_scene.moving_objects:
self.rman_scene.export_instances_motion()
@time_this
def update_scene(self, context, depsgraph):
#self.rman_updates = dict()
self.num_instances_changed = False
self.frame_number_changed = False
self.check_all_instances = False
self.rman_scene.depsgraph = depsgraph
self.rman_scene.bl_scene = depsgraph.scene
self.rman_scene.context = context
self.rman_scene.bl_view_layer = depsgraph.view_layer_eval
rfb_log().debug("------Start update scene--------")
# Check the number of instances. If we differ, an object may have been
# added or deleted
if self.rman_scene.num_object_instances != len(depsgraph.object_instances):
rfb_log().debug("\tNumber of instances changed: %d -> %d" % (self.rman_scene.num_object_instances, len(depsgraph.object_instances)))
self.num_instances_changed = True
self.rman_scene.num_object_instances = len(depsgraph.object_instances)
for dps_update in reversed(depsgraph.updates):
if isinstance(dps_update.id, bpy.types.Scene):
self.scene_updated()
elif isinstance(dps_update.id, bpy.types.World):
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
self.rman_scene.export_integrator()
self.rman_scene.export_samplefilters()
self.rman_scene.export_displayfilters()
self.rman_scene.export_viewport_stats()
elif isinstance(dps_update.id, bpy.types.Camera):
rfb_log().debug("Camera updated: %s" % dps_update.id.name)
if self.rman_scene.is_viewport_render:
if self.rman_scene.bl_scene.camera.data != dps_update.id:
continue
rman_sg_camera = self.rman_scene.main_camera
translator = self.rman_scene.rman_translators['CAMERA']
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
translator.update_viewport_cam(self.rman_scene.bl_scene.camera, rman_sg_camera, force_update=True)
else:
translator = self.rman_scene.rman_translators['CAMERA']
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
for ob, rman_sg_camera in self.rman_scene.rman_cameras.items():
if ob.original.name != dps_update.id.name:
continue
translator._update_render_cam(ob.original, rman_sg_camera)
elif isinstance(dps_update.id, bpy.types.Material):
rfb_log().debug("Material updated: %s" % dps_update.id.name)
self.material_updated(dps_update)
elif isinstance(dps_update.id, bpy.types.Mesh):
rfb_log().debug("Mesh updated: %s" % dps_update.id.name)
elif isinstance(dps_update.id, bpy.types.Light):
self.check_light_datablock(dps_update)
elif isinstance(dps_update.id, bpy.types.ParticleSettings):
self.check_particle_settings(dps_update)
elif isinstance(dps_update.id, bpy.types.ShaderNodeTree):
self.check_shader_nodetree(dps_update)
elif isinstance(dps_update.id, bpy.types.Object):
self.check_object_datablock(dps_update)
elif isinstance(dps_update.id, bpy.types.Collection):
rfb_log().debug("Collection updated: %s" % dps_update.id.name)
#self.update_collection(dps_update.id)
elif isinstance(dps_update.id, bpy.types.GeometryNodeTree):
# create an empty RmanUpdate
self.create_rman_update(dps_update.id.original, clear_instances=False)
else:
rfb_log().debug("Not handling %s update: %s" % (str(type(dps_update.id)), dps_update.id.name))
if not self.rman_updates and self.num_instances_changed:
# The number of instances changed, but we are not able
# to determine what changed. We are forced to check
# all instances.
rfb_log().debug("Set check_all_instances to True")
self.check_all_instances = True
if self.check_all_instances or self.rman_updates:
self.check_instances()
# call txmake all in case of new textures
texture_utils.get_txmanager().txmake_all(blocking=False)
self.rman_updates = dict()
rfb_log().debug("------End update scene----------")
@time_this
def check_instances(self, batch_mode=False):
deleted_obj_keys = list(self.rman_scene.rman_prototypes) # list of potential objects to delete
already_udpated = list() # list of objects already updated during our loop
clear_instances = list() # list of objects who had their instances cleared
rfb_log().debug("Updating instances")
with self.rman_scene.rman.SGManager.ScopedEdit(self.rman_scene.sg_scene):
for instance in self.rman_scene.depsgraph.object_instances:
if instance.object.type in ('CAMERA'):
continue
ob_key = instance.object.original
ob_eval = instance.object.evaluated_get(self.rman_scene.depsgraph)
instance_parent = None
psys = None
is_new_object = False
proto_key = object_utils.prototype_key(instance)
is_empty_instancer = False
is_instance = instance.is_instance
if is_instance:
ob_key = instance.instance_object.original
psys = instance.particle_system
instance_parent = instance.parent
is_empty_instancer = object_utils.is_empty_instancer(instance_parent)
if proto_key in deleted_obj_keys:
deleted_obj_keys.remove(proto_key)
rman_type = object_utils._detect_primitive_(ob_eval)
rman_sg_node = self.rman_scene.get_rman_prototype(proto_key)
if rman_sg_node and rman_type != rman_sg_node.rman_type:
# Types don't match
#
# This can happen because
# we have not been able to tag our types before Blender
# tells us an object has been added
# For now, just delete the existing sg_node
rfb_log().debug("\tTypes don't match. Removing: %s" % proto_key)
del self.rman_scene.rman_prototypes[proto_key]
rman_sg_node = None
rman_update = self.rman_updates.get(ob_key, None)
if not rman_sg_node:
# this is a new object.
if not self.rman_scene.check_visibility(instance, ob_eval=ob_eval):
# don't t export this object if it's not visible
continue
rman_sg_node = self.rman_scene.export_data_block(proto_key, ob_eval)
if not rman_sg_node:
continue
rfb_log().debug("\tNew Object added: %s (%s)" % (proto_key, rman_type))
if rman_type == 'LIGHTFILTER':
# update all lights with this light filter
users = bpy.context.blend_data.user_map(subset={ob_eval.original})
for o in users[ob_eval.original]:
if isinstance(o, bpy.types.Light):
o.node_tree.update_tag()
self.rman_scene.set_root_lightlinks() # update lightlinking on the root node
continue
is_new_object = True
if rman_update is None:
rman_update = self.create_rman_update(ob_key, update_geometry=False, update_shading=True, update_transform=True)
# set update_geometry to False
# since we've already exported the datablock
rman_update.is_updated_geometry = False
clear_instances.append(rman_sg_node)
if self.check_all_instances:
# check all instances in the scene
# build an RmanUpdate instance if it doesn't exist
if rman_update is None:
rman_update = RmanUpdate()
rman_update.is_updated_shading = True
rman_update.is_updated_transform = True
if rman_sg_node.is_frame_sensitive and self.frame_number_changed:
rman_update.is_updated_geometry = True
self.rman_updates[ob_key] = rman_update
elif rman_update is None:
# no RmanUpdate exists for this object
#
# check if one of the users of this object updated
# ex: the object was instanced via a GeometryNodeTree, and the
# geometry node tree updated
users = self.rman_scene.context.blend_data.user_map(subset={ob_eval.original})
user_exist = False
for o in users[ob_eval.original]:
if o.original in self.rman_updates:
parent = getattr(o.original, 'parent', None)
if parent == ob_eval.original:
# don't consider this object if there's a
# parent/child relation
continue
rfb_log().debug("\t%s user updated (%s)" % (ob_eval.name, o.name))
user_exist = True
break
if user_exist:
rman_update = self.create_rman_update(ob_key, update_transform=True)
else:
# check if the instance_parent was the thing that
# changed
if not instance_parent:
continue
rman_update = self.rman_updates.get(instance_parent.original, None)
if rman_update is None:
continue
if rman_update.is_updated_attributes:
continue
rfb_log().debug("\t%s parent updated (%s)" % (ob_eval.name, instance_parent.name))
# Originally, we were only updating the prototype, if the DepsgraphInstance
# was not is_instance. However, this doesn't work for META
# objects as the mesh has is_instance=True.
# So we now let whichever DepsgraphInstance we get first do the prototype updating
if rman_sg_node and not is_new_object: # and not is_instance:
if proto_key not in already_udpated:
if rman_update.is_updated_attributes:
# this should be a simple attribute change
from .rfb_utils import property_utils
attrs = rman_sg_node.sg_attributes.GetAttributes()
rm = ob_eval.renderman
if is_empty_instancer:
rm = instance_parent.renderman
meta = rm.prop_meta[rman_update.updated_prop_name]
rfb_log().debug("Setting RiAttribute: %s" % rman_update.updated_prop_name)
property_utils.set_riattr_bl_prop(attrs, rman_update.updated_prop_name, meta, rm, check_inherit=True)
rman_sg_node.sg_attributes.SetAttributes(attrs)
elif rman_update.is_updated_geometry:
translator = self.rman_scene.rman_translators.get(rman_type, None)
if rman_update.updated_prop_name:
rfb_log().debug("\tUpdating Single Primvar: %s" % proto_key)
translator.update_primvar(ob_eval, rman_sg_node, rman_update.updated_prop_name)
else:
rfb_log().debug("\tUpdating Object: %s" % proto_key)
translator.update(ob_eval, rman_sg_node)
#rman_sg_node.shared_attrs.Clear()
self.rman_scene.attach_material(ob_eval, rman_sg_node, sg_node=rman_sg_node.sg_attributes)
self.update_particle_emitters(ob_eval)
already_udpated.append(proto_key)
if rman_type in object_utils._RMAN_NO_INSTANCES_:
if rman_type == 'EMPTY':
self.rman_scene._export_hidden_instance(ob_eval, rman_sg_node)
continue
if rman_update.do_clear_instances:
# clear all instances for this prototype, if
# we have not already done so
if psys and instance_parent:
parent_proto_key = object_utils.prototype_key(instance_parent)
rman_parent_node = self.rman_scene.get_rman_prototype(parent_proto_key, ob=instance_parent)
if rman_parent_node and rman_parent_node not in clear_instances:
rfb_log().debug("\tClearing parent instances: %s" % parent_proto_key)
rman_parent_node.clear_instances()
clear_instances.append(rman_parent_node)
if rman_sg_node not in clear_instances:
rfb_log().debug("\tClearing instances: %s" % proto_key)
rman_sg_node.clear_instances()
clear_instances.append(rman_sg_node)
if not self.rman_scene.check_visibility(instance):
# This instance is not visible in the viewport. Don't
# add an instance of it
continue
self.rman_scene.export_instance(ob_eval, instance, rman_sg_node, rman_type, instance_parent, psys)
else:
if rman_sg_node not in clear_instances:
# this might be a bit werid, but we don't want another RmanUpdate
# instance to clear the instances afterwards, so we add to the
# clear_instances list
clear_instances.append(rman_sg_node)
# simply grab the existing instance and update the transform and/or material
rman_sg_group = self.rman_scene.get_rman_sg_instance(instance, rman_sg_node, instance_parent, psys, create=False)
rman_group_translator = self.rman_scene.rman_translators['GROUP']
if rman_sg_group:
# update instance attributes
self.rman_scene.update_instance_attributes(rman_group_translator, rman_sg_group, ob_eval, instance, remove=True)
if rman_update.is_updated_attributes:
continue
if rman_update.is_updated_transform:
rman_group_translator.update_transform(instance, rman_sg_group)
if is_empty_instancer and instance_parent.renderman.rman_material_override:
self.rman_scene.attach_material(instance_parent, rman_sg_group)
elif rman_update.is_updated_shading:
if psys:
self.rman_scene.attach_particle_material(psys.settings, instance_parent, ob_eval, rman_sg_group)
else:
self.rman_scene.attach_material(ob_eval, rman_sg_group)
else:
# Didn't get an rman_sg_group. Do a full instance export.
self.rman_scene.export_instance(ob_eval, instance, rman_sg_node, rman_type, instance_parent, psys)
if not batch_mode:
if rman_type == 'LIGHT':
# We are dealing with a light. Check if it's a solo light, or muted
self.rman_scene.check_solo_light(rman_sg_node, ob_eval)
# check portal lights
self.update_portals(ob_eval)
# Hide the default light
if self.rman_scene.default_light.GetHidden() != 1:
self.rman_scene.default_light.SetHidden(1)
# Delete any removed partcle systems
if proto_key in self.rman_scene.rman_particles:
ob_psys = self.rman_scene.rman_particles[proto_key]
rman_particle_nodes = list(ob_psys)
for psys in ob_eval.particle_systems:
try:
rman_particle_nodes.remove(psys.settings.original)
except:
continue
if rman_particle_nodes:
rfb_log().debug("\t\tRemoving particle nodes: %s" % proto_key)
for k in rman_particle_nodes:
del ob_psys[k]