| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2019-02-18 07:21:50 +11:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2011, Blender Foundation. | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 11:28:09 +01:00
										 |  |  | #include <cmath>
 | 
					
						
							| 
									
										
										
										
											2020-09-04 16:23:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | #include "COM_WrapOperation.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-19 18:13:26 +01:00
										 |  |  | WrapOperation::WrapOperation(DataType datatype) : ReadBufferOperation(datatype) | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |   this->m_wrappingType = CMP_NODE_WRAP_NONE; | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline float WrapOperation::getWrappedOriginalXPos(float x) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-23 11:21:22 +10:00
										 |  |  |   if (this->getWidth() == 0) { | 
					
						
							| 
									
										
										
										
											2013-07-03 15:33:14 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2019-04-23 11:21:22 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   while (x < 0) { | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |     x += this->m_width; | 
					
						
							| 
									
										
										
										
											2019-04-23 11:21:22 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |   return fmodf(x, this->getWidth()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline float WrapOperation::getWrappedOriginalYPos(float y) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-04-23 11:21:22 +10:00
										 |  |  |   if (this->getHeight() == 0) { | 
					
						
							| 
									
										
										
										
											2013-07-03 15:33:14 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2019-04-23 11:21:22 +10:00
										 |  |  |   } | 
					
						
							|  |  |  |   while (y < 0) { | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |     y += this->m_height; | 
					
						
							| 
									
										
										
										
											2019-04-23 11:21:22 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |   return fmodf(y, this->getHeight()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-19 11:06:16 +01:00
										 |  |  | void WrapOperation::executePixelSampled(float output[4], float x, float y, PixelSampler sampler) | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   float nx, ny; | 
					
						
							|  |  |  |   nx = x; | 
					
						
							|  |  |  |   ny = y; | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |   MemoryBufferExtend extend_x = COM_MB_CLIP, extend_y = COM_MB_CLIP; | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |   switch (m_wrappingType) { | 
					
						
							|  |  |  |     case CMP_NODE_WRAP_NONE: | 
					
						
							| 
									
										
										
										
											2019-05-01 10:50:02 +10:00
										 |  |  |       // Intentionally empty, originalXPos and originalYPos have been set before
 | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case CMP_NODE_WRAP_X: | 
					
						
							|  |  |  |       // wrap only on the x-axis
 | 
					
						
							|  |  |  |       nx = this->getWrappedOriginalXPos(x); | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |       extend_x = COM_MB_REPEAT; | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case CMP_NODE_WRAP_Y: | 
					
						
							|  |  |  |       // wrap only on the y-axis
 | 
					
						
							|  |  |  |       ny = this->getWrappedOriginalYPos(y); | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |       extend_y = COM_MB_REPEAT; | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case CMP_NODE_WRAP_XY: | 
					
						
							|  |  |  |       // wrap on both
 | 
					
						
							|  |  |  |       nx = this->getWrappedOriginalXPos(x); | 
					
						
							|  |  |  |       ny = this->getWrappedOriginalYPos(y); | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |       extend_x = COM_MB_REPEAT; | 
					
						
							|  |  |  |       extend_y = COM_MB_REPEAT; | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |   executePixelExtend(output, nx, ny, sampler, extend_x, extend_y); | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool WrapOperation::determineDependingAreaOfInterest(rcti *input, | 
					
						
							|  |  |  |                                                      ReadBufferOperation *readOperation, | 
					
						
							|  |  |  |                                                      rcti *output) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   rcti newInput; | 
					
						
							|  |  |  |   newInput.xmin = input->xmin; | 
					
						
							|  |  |  |   newInput.xmax = input->xmax; | 
					
						
							|  |  |  |   newInput.ymin = input->ymin; | 
					
						
							|  |  |  |   newInput.ymax = input->ymax; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 12:30:59 +11:00
										 |  |  |   if (ELEM(m_wrappingType, CMP_NODE_WRAP_X, CMP_NODE_WRAP_XY)) { | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |     // wrap only on the x-axis if tile is wrapping
 | 
					
						
							|  |  |  |     newInput.xmin = getWrappedOriginalXPos(input->xmin); | 
					
						
							| 
									
										
										
										
											2020-09-04 16:23:00 +02:00
										 |  |  |     newInput.xmax = roundf(getWrappedOriginalXPos(input->xmax)); | 
					
						
							| 
									
										
										
										
											2014-06-30 18:22:27 +06:00
										 |  |  |     if (newInput.xmin >= newInput.xmax) { | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |       newInput.xmin = 0; | 
					
						
							|  |  |  |       newInput.xmax = this->getWidth(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-11-06 12:30:59 +11:00
										 |  |  |   if (ELEM(m_wrappingType, CMP_NODE_WRAP_Y, CMP_NODE_WRAP_XY)) { | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |     // wrap only on the y-axis if tile is wrapping
 | 
					
						
							|  |  |  |     newInput.ymin = getWrappedOriginalYPos(input->ymin); | 
					
						
							| 
									
										
										
										
											2020-09-04 16:23:00 +02:00
										 |  |  |     newInput.ymax = roundf(getWrappedOriginalYPos(input->ymax)); | 
					
						
							| 
									
										
										
										
											2014-06-30 18:22:27 +06:00
										 |  |  |     if (newInput.ymin >= newInput.ymax) { | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  |       newInput.ymin = 0; | 
					
						
							|  |  |  |       newInput.ymax = this->getHeight(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-05 10:45:21 +00:00
										 |  |  |   return ReadBufferOperation::determineDependingAreaOfInterest(&newInput, readOperation, output); | 
					
						
							| 
									
										
										
										
											2013-02-06 08:40:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void WrapOperation::setWrapping(int wrapping_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   m_wrappingType = wrapping_type; | 
					
						
							|  |  |  | } |