|
|
|
@ -244,6 +244,179 @@ void video_format_ScaleCropAr( video_format_t *p_dst, const video_format_t *p_sr |
|
|
|
p_dst->i_sar_num, p_dst->i_sar_den, 65536); |
|
|
|
} |
|
|
|
|
|
|
|
//Simplify transforms to have something more managable. Order: angle, hflip.
|
|
|
|
void transform_GetBasicOps( video_transform_t transform, int *angle, int *hflip ) { |
|
|
|
|
|
|
|
*angle = 0; |
|
|
|
*hflip = 0; |
|
|
|
|
|
|
|
switch ( transform ) { |
|
|
|
|
|
|
|
case TRANSFORM_R90: |
|
|
|
*angle = 90; |
|
|
|
break; |
|
|
|
case TRANSFORM_R180: |
|
|
|
*angle = 180; |
|
|
|
break; |
|
|
|
case TRANSFORM_R270: |
|
|
|
*angle = 270; |
|
|
|
break; |
|
|
|
case TRANSFORM_HFLIP: |
|
|
|
*hflip = 1; |
|
|
|
break; |
|
|
|
case TRANSFORM_VFLIP: |
|
|
|
*angle = 180; |
|
|
|
*hflip = 1; |
|
|
|
break; |
|
|
|
case TRANSFORM_TRANSPOSE: |
|
|
|
*angle = 90; |
|
|
|
*hflip = 1; |
|
|
|
break; |
|
|
|
case TRANSFORM_ANTI_TRANSPOSE: |
|
|
|
*angle = 270; |
|
|
|
*hflip = 1; |
|
|
|
break; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
video_transform_t transform_FromBasicOps( int angle, int hflip ) |
|
|
|
{ |
|
|
|
switch ( angle ) { |
|
|
|
case 0: |
|
|
|
return hflip ? TRANSFORM_HFLIP : TRANSFORM_IDENTIY; |
|
|
|
case 90: |
|
|
|
return hflip ? TRANSFORM_TRANSPOSE : TRANSFORM_R90; |
|
|
|
case 180: |
|
|
|
return hflip ? TRANSFORM_VFLIP : TRANSFORM_R180; |
|
|
|
case 270: |
|
|
|
return hflip ? TRANSFORM_ANTI_TRANSPOSE : TRANSFORM_R270; |
|
|
|
default: |
|
|
|
return TRANSFORM_IDENTIY; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
video_transform_t video_format_GetTransform( video_orientation_t src, video_orientation_t dst ) |
|
|
|
{ |
|
|
|
int angle1 = 0; |
|
|
|
int hflip1 = 0; |
|
|
|
|
|
|
|
transform_GetBasicOps( (video_transform_t)src, &angle1, &hflip1 ); |
|
|
|
|
|
|
|
int angle2 = 0; |
|
|
|
int hflip2 = 0; |
|
|
|
|
|
|
|
transform_GetBasicOps( transform_Inverse( (video_transform_t)dst ), &angle2, &hflip2 ); |
|
|
|
|
|
|
|
int angle = (angle1 + angle2) % 360; |
|
|
|
int hflip = (hflip1 + hflip2) % 2; |
|
|
|
|
|
|
|
return transform_FromBasicOps(angle, hflip); |
|
|
|
} |
|
|
|
|
|
|
|
void video_format_TransformTo( video_format_t *fmt, video_orientation_t dst_orientation ) |
|
|
|
{ |
|
|
|
video_transform_t transform = video_format_GetTransform(fmt->orientation, dst_orientation); |
|
|
|
|
|
|
|
video_format_TransformBy(fmt, transform); |
|
|
|
} |
|
|
|
|
|
|
|
void video_format_TransformBy( video_format_t *fmt, video_transform_t transform ) |
|
|
|
{ |
|
|
|
/* Get destination orientation */ |
|
|
|
|
|
|
|
int angle1 = 0; |
|
|
|
int hflip1 = 0; |
|
|
|
|
|
|
|
transform_GetBasicOps( transform, &angle1, &hflip1 ); |
|
|
|
|
|
|
|
int angle2 = 0; |
|
|
|
int hflip2 = 0; |
|
|
|
|
|
|
|
transform_GetBasicOps( (video_transform_t)( fmt->orientation ), &angle2, &hflip2 ); |
|
|
|
|
|
|
|
int angle = (angle2 - angle1 + 360) % 360; |
|
|
|
int hflip = (hflip2 - hflip1 + 2) % 2; |
|
|
|
|
|
|
|
video_orientation_t dst_orient = ORIENT_NORMAL; |
|
|
|
|
|
|
|
if( hflip ) { |
|
|
|
|
|
|
|
if( angle == 0 ) |
|
|
|
dst_orient = ORIENT_HFLIPPED; |
|
|
|
else if( angle == 90 ) |
|
|
|
dst_orient = ORIENT_ANTI_TRANSPOSED; |
|
|
|
else if( angle == 180 ) |
|
|
|
dst_orient = ORIENT_VFLIPPED; |
|
|
|
else if( angle == 270 ) |
|
|
|
dst_orient = ORIENT_TRANSPOSED; |
|
|
|
} |
|
|
|
else { |
|
|
|
|
|
|
|
if( angle == 90 ) |
|
|
|
dst_orient = ORIENT_ROTATED_90; |
|
|
|
else if( angle == 180 ) |
|
|
|
dst_orient = ORIENT_ROTATED_180; |
|
|
|
else if( angle == 270 ) |
|
|
|
dst_orient = ORIENT_ROTATED_270; |
|
|
|
} |
|
|
|
|
|
|
|
/* Apply transform */ |
|
|
|
|
|
|
|
video_format_t scratch = *fmt; |
|
|
|
|
|
|
|
if( ORIENT_IS_SWAP( fmt->orientation ) != ORIENT_IS_SWAP( dst_orient )) { |
|
|
|
|
|
|
|
fmt->i_width = scratch.i_height; |
|
|
|
fmt->i_visible_width = scratch.i_visible_height; |
|
|
|
fmt->i_height = scratch.i_width; |
|
|
|
fmt->i_visible_height = scratch.i_visible_width; |
|
|
|
fmt->i_sar_num = scratch.i_sar_den; |
|
|
|
fmt->i_sar_den = scratch.i_sar_num; |
|
|
|
} |
|
|
|
|
|
|
|
unsigned int delta_y = scratch.i_height - scratch.i_visible_height - scratch.i_y_offset; |
|
|
|
unsigned int delta_x = scratch.i_width - scratch.i_visible_width - scratch.i_x_offset; |
|
|
|
|
|
|
|
switch ( transform ) |
|
|
|
{ |
|
|
|
case TRANSFORM_R90: |
|
|
|
fmt->i_x_offset = delta_y; |
|
|
|
fmt->i_y_offset = scratch.i_x_offset; |
|
|
|
break; |
|
|
|
case TRANSFORM_R180: |
|
|
|
fmt->i_x_offset = delta_x; |
|
|
|
fmt->i_y_offset = delta_y; |
|
|
|
break; |
|
|
|
case TRANSFORM_R270: |
|
|
|
fmt->i_x_offset = scratch.i_y_offset; |
|
|
|
fmt->i_y_offset = delta_x; |
|
|
|
break; |
|
|
|
case TRANSFORM_HFLIP: |
|
|
|
fmt->i_x_offset = delta_x; |
|
|
|
break; |
|
|
|
case TRANSFORM_VFLIP: |
|
|
|
fmt->i_y_offset = delta_y; |
|
|
|
break; |
|
|
|
case TRANSFORM_TRANSPOSE: |
|
|
|
fmt->i_x_offset = scratch.i_y_offset; |
|
|
|
fmt->i_y_offset = scratch.i_x_offset; |
|
|
|
break; |
|
|
|
case TRANSFORM_ANTI_TRANSPOSE: |
|
|
|
fmt->i_x_offset = delta_y; |
|
|
|
fmt->i_y_offset = delta_x; |
|
|
|
break; |
|
|
|
} |
|
|
|
|
|
|
|
fmt->orientation = dst_orient; |
|
|
|
} |
|
|
|
|
|
|
|
void video_format_ApplyRotation( const video_format_t * restrict in, video_format_t * restrict out ) |
|
|
|
{ |
|
|
|
*out = *in; |
|
|
|
|
|
|
|
video_format_TransformTo(out, ORIENT_NORMAL); |
|
|
|
} |
|
|
|
|
|
|
|
bool video_format_IsSimilar( const video_format_t *p_fmt1, const video_format_t *p_fmt2 ) |
|
|
|
{ |
|
|
|
video_format_t v1 = *p_fmt1; |
|
|
|
|