robust attribute copy for duplicate frames
This commit is contained in:
		
							parent
							
								
									6dbf666ee3
								
							
						
					
					
						commit
						edfefa874a
					
				
							
								
								
									
										163
									
								
								utils.py
									
									
									
									
									
								
							
							
						
						
									
										163
									
								
								utils.py
									
									
									
									
									
								
							| @ -590,8 +590,9 @@ def copy_stroke_to_frame(s, frame, select=True): | |||||||
|     ns.points.update() |     ns.points.update() | ||||||
|     return ns |     return ns | ||||||
| 
 | 
 | ||||||
| def bulk_copy_attributes(source_attr, target_attr): | """## Works, but do not copy all attributes type (probably ok for GP though) | ||||||
|     # Get the data as flat numpy array based on attribute type and domain | def bulk_frame_copy_attributes(source_attr, target_attr): | ||||||
|  |     '''Get and apply data as flat numpy array based on attribute type''' | ||||||
|     if source_attr.data_type == 'INT': |     if source_attr.data_type == 'INT': | ||||||
|         data = np.empty(len(source_attr.data), dtype=np.int32) |         data = np.empty(len(source_attr.data), dtype=np.int32) | ||||||
|         source_attr.data.foreach_get('value', data) |         source_attr.data.foreach_get('value', data) | ||||||
| @ -617,62 +618,105 @@ def bulk_copy_attributes(source_attr, target_attr): | |||||||
|         source_attr.data.foreach_get('value', data) |         source_attr.data.foreach_get('value', data) | ||||||
|         target_attr.data.foreach_set('value', data) |         target_attr.data.foreach_set('value', data) | ||||||
| 
 | 
 | ||||||
|     ## Filter by domain ? (not needed it seem, dtypes are the same) | ## works in slowmotion (keep as reference for testing) | ||||||
|     ''' | # def copy_attribute_values(src_dr, dst_dr, source_attr, dest_attr, data_size): | ||||||
|     if source_attr.domain == 'POINT': | #     ## Zip method to copy one by one | ||||||
|         if source_attr.data_type == 'FLOAT': | #     val_type = {'FLOAT_COLOR': 'color','FLOAT_VECTOR': 'vector'}.get(source_attr.data_type, 'value') | ||||||
|             data = np.empty(len(source_attr.data), dtype=np.float32) | #     src_start = src_dr.curve_offsets[0].value | ||||||
|             source_attr.data.foreach_get('value', data) | #     src_end = src_start + data_size  | ||||||
|             target_attr.data.foreach_set('value', data) | #     dst_start = dst_dr.curve_offsets[0].value | ||||||
|         elif source_attr.data_type == 'FLOAT_VECTOR': | #     dst_end = dst_start + data_size | ||||||
|             data = np.empty(len(source_attr.data) * 3, dtype=np.float32) | #     for src_idx, dest_idx in zip(range(src_start, src_end),range(dst_start, dst_end)): | ||||||
|             source_attr.data.foreach_get('vector', data) | #         setattr(dest_attr.data[dest_idx], val_type, getattr(source_attr.data[src_idx], val_type)) | ||||||
|             target_attr.data.foreach_set('vector', data) | """ | ||||||
|         elif source_attr.data_type == 'FLOAT_COLOR': | 
 | ||||||
|             data = np.empty(len(source_attr.data) * 4, dtype=np.float32) | attribute_value_string = { | ||||||
|             source_attr.data.foreach_get('color', data) |   'FLOAT': "value", | ||||||
|             target_attr.data.foreach_set('color', data) |   'INT': "value", | ||||||
|         ## No Boolean by defaut on points domain |   'FLOAT_VECTOR': "vector", | ||||||
|     elif source_attr.domain == 'CURVE': |   'FLOAT_COLOR': "color", | ||||||
|         if source_attr.data_type == 'INT': |   'BYTE_COLOR': "color", | ||||||
|             data = np.empty(len(source_attr.data), dtype=np.int32) |   'STRING': "value", | ||||||
|             source_attr.data.foreach_get('value', data) |   'BOOLEAN': "value", | ||||||
|             target_attr.data.foreach_set('value', data) |   'FLOAT2': "value", | ||||||
|         elif source_attr.data_type == 'INT8': |   'INT8': "value", | ||||||
|             data = np.empty(len(source_attr.data), dtype=np.int8) |   'INT32_2D': "value", | ||||||
|             source_attr.data.foreach_get('value', data) |   'QUATERNION': "value", | ||||||
|             target_attr.data.foreach_set('value', data) |   'FLOAT4X4': "value", | ||||||
|         elif source_attr.data_type == 'FLOAT_COLOR': | } | ||||||
|             data = np.empty(len(source_attr.data) * 4, dtype=np.float32) | 
 | ||||||
|             source_attr.data.foreach_get('color', data) | attribute_value_dtype = { | ||||||
|             target_attr.data.foreach_set('color', data) |   'FLOAT': np.float32, | ||||||
|         elif source_attr.data_type == 'BOOLEAN': |   'INT': np.dtype('int'), | ||||||
|             data = np.empty(len(source_attr.data), dtype=bool) |   'FLOAT_VECTOR': np.float32, | ||||||
|             source_attr.data.foreach_get('value', data) |   'FLOAT_COLOR': np.float32, | ||||||
|             target_attr.data.foreach_set('value', data) |   'BYTE_COLOR': np.int8, | ||||||
|     ''' |   'STRING': np.dtype('str'), | ||||||
|  |   'BOOLEAN': np.dtype('bool'), | ||||||
|  |   'FLOAT2': np.float32, | ||||||
|  |   'INT8': np.int8, | ||||||
|  |   'INT32_2D': np.dtype('int'), | ||||||
|  |   'QUATERNION': np.float32, | ||||||
|  |   'FLOAT4X4': np.float32, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | attribute_value_shape = { | ||||||
|  |   'FLOAT': (), | ||||||
|  |   'INT': (), | ||||||
|  |   'FLOAT_VECTOR': (3,), | ||||||
|  |   'FLOAT_COLOR': (4,), | ||||||
|  |   'BYTE_COLOR': (4,), | ||||||
|  |   'STRING': (), | ||||||
|  |   'BOOLEAN': (), | ||||||
|  |   'FLOAT2':(2,), | ||||||
|  |   'INT8': (), | ||||||
|  |   'INT32_2D': (2,), | ||||||
|  |   'QUATERNION': (4,), | ||||||
|  |   'FLOAT4X4': (4,4), | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def copy_attribute_values(src_dr, dst_dr, source_attr, dest_attr): | def bulk_copy_attributes(source_attr, target_attr): | ||||||
|     ## Zip method to copy one by one |     '''Get and apply data as flat numpy array based on attribute type''' | ||||||
|     ## One liner |     value_string = attribute_value_string[source_attr.data_type] | ||||||
|     # val_type = {'FLOAT_COLOR': 'color','FLOAT_VECTOR': 'vector'}.get(source_attr.data_type, 'value') |     dtype = attribute_value_dtype[source_attr.data_type] | ||||||
|     if source_attr.data_type == 'FLOAT_COLOR': |     shape = attribute_value_shape[source_attr.data_type] | ||||||
|         val_type = 'color' | 
 | ||||||
|     elif source_attr.data_type == 'FLOAT_VECTOR': |     domain_size = len(source_attr.data) | ||||||
|         val_type = 'vector' |     ## Need to pass attributes to get domain size | ||||||
|     else: |     # domain_size = attributes.domain_size(source_attr.domain) | ||||||
|         val_type = 'value' | 
 | ||||||
|     ## /!\ DOESN'T WORK. index is wrong |     # start = time() | ||||||
|     for src_idx, dest_idx in zip( |     data = np.empty((domain_size, *shape), dtype=dtype).ravel() | ||||||
|             range(src_dr.curve_offsets[0].value, src_dr.curve_offsets[-1].value), |     source_attr.data.foreach_get(value_string, data) | ||||||
|             range(dst_dr.curve_offsets[0].value, dst_dr.curve_offsets[-1].value)): |     target_attr.data.foreach_set(value_string, data) | ||||||
|         setattr(dest_attr.data[dest_idx], val_type, getattr(source_attr.data[src_idx], val_type)) |     # end = time() | ||||||
|  |     # np_empty = end - start | ||||||
|  | 
 | ||||||
|  |     ## np.prod (works, supposedly faster but tested slower) | ||||||
|  |     # data = np.empty(int(domain_size * np.prod(shape)), dtype=dtype) | ||||||
|  |     # source_attr.data.foreach_get(value_string, data) | ||||||
|  |     # target_attr.data.foreach_set(value_string, data) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     ## np.zeros (works, sometimes faster on big set of attributes) | ||||||
|  |     # start = time() | ||||||
|  |     # data = np.zeros((domain_size, *shape), dtype=dtype) | ||||||
|  |     # source_attr.data.foreach_get(value_string, np.ravel(data)) | ||||||
|  |     # target_attr.data.foreach_set(value_string, np.ravel(data)) | ||||||
|  |     # end = time() | ||||||
|  |     # np_zero = end - start | ||||||
|  |      | ||||||
|  |     # print('np EMPTY faster' if np_empty < np_zero else 'np ZERO faster', source_attr.domain, source_attr.data_type, domain_size) | ||||||
|  |     # print('np_zero', np_zero) | ||||||
|  |     # print('np_empty', np_empty) | ||||||
|  |     # print() | ||||||
| 
 | 
 | ||||||
| def copy_frame_at(source_frame, layer, frame_number): | def copy_frame_at(source_frame, layer, frame_number): | ||||||
|     '''Copy a frame (source_frame) to a layer at given frame_number''' |     '''Copy a frame (source_frame) to a layer at given frame_number''' | ||||||
|     source_drawing = source_frame.drawing |     source_drawing = source_frame.drawing | ||||||
| 
 | 
 | ||||||
|  |     # frame_copy_start = time() # time_dbg | ||||||
|     frame = layer.frames.new(frame_number) |     frame = layer.frames.new(frame_number) | ||||||
|     dr = frame.drawing |     dr = frame.drawing | ||||||
|     dr.add_strokes([len(s.points) for s in source_drawing.strokes]) |     dr.add_strokes([len(s.points) for s in source_drawing.strokes]) | ||||||
| @ -682,13 +726,18 @@ def copy_frame_at(source_frame, layer, frame_number): | |||||||
|             dr.attributes.new( |             dr.attributes.new( | ||||||
|                 name=attr_name, type=source_attr.data_type, domain=source_attr.domain) |                 name=attr_name, type=source_attr.data_type, domain=source_attr.domain) | ||||||
|         target_attr = dr.attributes[attr_name] |         target_attr = dr.attributes[attr_name] | ||||||
|         bulk_copy_attributes(source_attr, target_attr) |  | ||||||
| 
 | 
 | ||||||
|         ## Broken |         # start_time = time() # time_dbg-per-attrib | ||||||
|         # start_time = time() | 
 | ||||||
|         # copy_attribute_values(source_drawing, dr, source_attr, target_attr) |         # bulk_frame_copy_attributes(source_attr, target_attr) # only some attributes | ||||||
|         # end_time = time() |         bulk_copy_attributes(source_attr, target_attr) | ||||||
|         # print(f"copy_attribute_values execution time: {end_time - start_time} seconds") |         # copy_attribute_values(source_drawing, dr, source_attr, target_attr, source_drawing.attributes.domain_size(source_attr.domain)) # super slow | ||||||
|  | 
 | ||||||
|  |         # end_time = time() # time_dbg-per-attrib | ||||||
|  |         # print(f"copy_attribute '{attr_name}' execution time: {end_time - start_time} seconds") # time_dbg-per-attrib | ||||||
|  | 
 | ||||||
|  |     # frame_copy_end = time() # time_dbg | ||||||
|  |     # print(f"frame copy execution time: {frame_copy_end - frame_copy_start} seconds") # time_dbg | ||||||
| 
 | 
 | ||||||
| # ----------------- | # ----------------- | ||||||
| ### Vector utils 3d | ### Vector utils 3d | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user