diff options
author | Ilia Alshanetsky <iliaa@php.net> | 2002-10-29 01:15:44 +0000 |
---|---|---|
committer | Ilia Alshanetsky <iliaa@php.net> | 2002-10-29 01:15:44 +0000 |
commit | e34ea3adb4ce3cba6bad97196add3f29be9834da (patch) | |
tree | 5a1f40dd10b8465d758eaec081f6593bdc6b6715 | |
parent | c82799be3206a51cd77964f715be3c1ba52b8569 (diff) | |
download | php-git-e34ea3adb4ce3cba6bad97196add3f29be9834da.tar.gz |
Added a patch by Pierre-Alain Joye <paj@pearfr.org>, which implements the
imagerotate() that allows rotation of images in gd.
-rw-r--r-- | ext/gd/gd.c | 37 | ||||
-rw-r--r-- | ext/gd/libgd/gd.c | 433 | ||||
-rw-r--r-- | ext/gd/libgd/gd.h | 6 | ||||
-rw-r--r-- | ext/gd/php_gd.h | 5 |
4 files changed, 481 insertions, 0 deletions
diff --git a/ext/gd/gd.c b/ext/gd/gd.c index 7cef7c0d2a..f56aa1c17f 100644 --- a/ext/gd/gd.c +++ b/ext/gd/gd.c @@ -158,6 +158,10 @@ function_entry gd_functions[] = { PHP_FE(imagecopyresampled, NULL) #endif +#ifdef HAVE_GD_BUNDLED + PHP_FE(imagerotate, NULL) +#endif + #if HAVE_GD_IMAGESETTILE PHP_FE(imagesettile, NULL) #endif @@ -912,6 +916,39 @@ PHP_FUNCTION(imagecopyresampled) gdImageCopyResampled(im_dst, im_src, dstX, dstY, srcX, srcY, dstW, dstH, srcW, srcH); RETURN_TRUE; + +} +/* }}} */ +#endif + +#ifdef HAVE_GD_BUNDLED +/* {{{ proto int imagerotate(int src_im, float angle, int bgdcolor) + Rotate an image using a custom angle */ +PHP_FUNCTION(imagerotate) +{ + zval **SIM, **ANGLE, **BGDCOLOR; + gdImagePtr im_dst, im_src; + double degrees; + long color; + + if (ZEND_NUM_ARGS() != 3 || zend_get_parameters_ex(3, &SIM, &ANGLE, &BGDCOLOR) == FAILURE) { + ZEND_WRONG_PARAM_COUNT(); + } + + ZEND_FETCH_RESOURCE(im_src, gdImagePtr, SIM, -1, "Image", le_gd); + + convert_to_long_ex(BGDCOLOR); + color = Z_LVAL_PP(BGDCOLOR); + + convert_to_double_ex(ANGLE); + degrees = Z_DVAL_PP(ANGLE); + im_dst = gdImageRotate(im_src, degrees, color); + + if (im_dst != NULL) { + ZEND_REGISTER_RESOURCE(return_value, im_dst, le_gd); + } else { + RETURN_FALSE; + } } /* }}} */ #endif diff --git a/ext/gd/libgd/gd.c b/ext/gd/libgd/gd.c index d09b61f537..7f40749fd7 100644 --- a/ext/gd/libgd/gd.c +++ b/ext/gd/libgd/gd.c @@ -2191,6 +2191,439 @@ gdImageCopyResampled (gdImagePtr dst, } } + +#ifdef ROTATE_PI +#undef ROTATE_PI +#endif /* ROTATE_PI */ + +#define ROTATE_DEG2RAD 3.1415926535897932384626433832795/180 +void gdImageSkewX (gdImagePtr dst, gdImagePtr src, int uRow, int iOffset, double dWeight, int clrBack) +{ + typedef int (*FuncPtr)(gdImagePtr, int, int); + int i, r, g, b, a; + FuncPtr f; + + int pxlOldLeft, pxlLeft, pxlSrc; + + if (src->trueColor) { + f = gdImageGetTrueColorPixel; + } else { + f = gdImageGetPixel; + } + + for (i = 0; i < iOffset; i++) { + gdImageSetPixel (dst, i, uRow, clrBack); + } + + if (i < dst->sx) { + gdImageSetPixel (dst, i, uRow, pxlLeft); + } + + pxlOldLeft = clrBack; + + for (i = 0; i < src->sx; i++) { + pxlSrc = f (src,i,uRow); + + r = gdImageRed(src,pxlSrc) * dWeight; + g = gdImageGreen(src,pxlSrc) * dWeight; + b = gdImageBlue(src,pxlSrc) * dWeight; + a = gdImageAlpha(src,pxlSrc) * dWeight; + + pxlLeft = gdImageColorAllocateAlpha(src, r, g, b, a); + + if (pxlLeft == -1) { + pxlLeft = gdImageColorClosestAlpha(src, r, g, b, a); + } + + r = gdImageRed(src,pxlSrc) - (gdImageRed(src,pxlLeft) - gdImageRed(src,pxlOldLeft)); + g = gdImageGreen(src,pxlSrc) - (gdImageGreen(src,pxlLeft) - gdImageGreen(src,pxlOldLeft)); + b = gdImageBlue(src,pxlSrc) - (gdImageBlue(src,pxlLeft) - gdImageBlue(src,pxlOldLeft)); + a = gdImageAlpha(src,pxlSrc) - (gdImageAlpha(src,pxlLeft) - gdImageAlpha(src,pxlOldLeft)); + + if (r>255) { + r = 255; + } + + if (g>255) { + g = 255; + } + + if(b>255) { + b = 255; + } + + if (a>127) { + b = 127; + } + + pxlSrc = gdImageColorAllocateAlpha(dst, r, g, b, a); + + if (pxlSrc == -1) { + pxlSrc = gdImageColorClosestAlpha(dst, r, g, b, a); + } + + if ((i + iOffset >= 0) && (i + iOffset < dst->sx)) { + gdImageSetPixel (dst, i+iOffset, uRow, pxlSrc); + } + + pxlOldLeft = pxlLeft; + } + + i += iOffset; + + if (i < dst->sx) { + gdImageSetPixel (dst, i, uRow, pxlLeft); + } + + gdImageSetPixel (dst, iOffset, uRow, clrBack); + + i--; + + while (++i < dst->sx) { + gdImageSetPixel (dst, i, uRow, clrBack); + } +} + +void gdImageSkewY (gdImagePtr dst, gdImagePtr src, int uCol, int iOffset, double dWeight, int clrBack) +{ + typedef int (*FuncPtr)(gdImagePtr, int, int); + int i, iYPos, r, g, b, a; + FuncPtr f; + int pxlOldLeft, pxlLeft, pxlSrc; + + if (src->trueColor) { + f = gdImageGetTrueColorPixel; + } else { + f = gdImageGetPixel; + } + + for (i = 0; i<iOffset; i++) { + gdImageSetPixel (dst, uCol, i, clrBack); + } + + pxlOldLeft = clrBack; + + for (i = 0; i < src->sy; i++) { + pxlSrc = f (src, uCol, i); + iYPos = i + iOffset; + + r = gdImageRed(src,pxlSrc) * dWeight; + g = gdImageGreen(src,pxlSrc) * dWeight; + b = gdImageBlue(src,pxlSrc) * dWeight; + a = gdImageAlpha(src,pxlSrc) * dWeight; + + pxlLeft = gdImageColorAllocateAlpha(src, r, g, b, a); + + if (pxlLeft == -1) { + pxlLeft = gdImageColorClosestAlpha(src, r, g, b, a); + } + + r = gdImageRed(src,pxlSrc) - (gdImageRed(src,pxlLeft) - gdImageRed(src,pxlOldLeft)); + g = gdImageGreen(src,pxlSrc) - (gdImageGreen(src,pxlLeft) - gdImageGreen(src,pxlOldLeft)); + b = gdImageBlue(src,pxlSrc) - (gdImageBlue(src,pxlLeft) - gdImageBlue(src,pxlOldLeft)); + a = gdImageAlpha(src,pxlSrc) - (gdImageAlpha(src,pxlLeft) - gdImageAlpha(src,pxlOldLeft)); + + if (r>255) { + r = 255; + } + + if (g>255) { + g = 255; + } + + if(b>255) { + b = 255; + } + + if (a>127) { + b = 127; + } + + pxlSrc = gdImageColorAllocateAlpha(dst, r, g, b, a); + + if (pxlSrc == -1) { + pxlSrc = gdImageColorClosestAlpha(dst, r, g, b, a); + } + + if ((iYPos >= 0) && (iYPos < dst->sy)) { + gdImageSetPixel (dst, uCol, iYPos, pxlSrc); + } + + pxlOldLeft = pxlLeft; + } + + i = iYPos; + if (i < dst->sy) { + gdImageSetPixel (dst, uCol, i, pxlLeft); + } + + i--; + while (++i < dst->sy) { + gdImageSetPixel (dst, uCol, i, clrBack); + } +} + +/* Rotates an image by 90 degrees (counter clockwise) */ +gdImagePtr gdImageRotate90 (gdImagePtr src) +{ + int uY, uX; + int c; + gdImagePtr dst; + typedef int (*FuncPtr)(gdImagePtr, int, int); + FuncPtr f; + + if (src->trueColor) { + dst = gdImageCreateTrueColor ( src->sx,src->sy); + f = gdImageGetTrueColorPixel; + } else { + dst = gdImageCreate (src->sx, src->sy); + f = gdImageGetPixel; + } + + if (dst != NULL) { + gdImagePaletteCopy (dst, src); + + for (uY = 0; uY<src->sy; uY++) { + for (uX = 0; uX<src->sx; uX++) { + c = f (src, uX, uY); + gdImageSetPixel(dst, uY, (dst->sy - uX - 1), c); + } + } + } + + return dst; +} + +/* Rotates an image by 180 degrees (counter clockwise) */ +gdImagePtr gdImageRotate180 (gdImagePtr src) +{ + int uY, uX; + int c; + gdImagePtr dst; + typedef int (*FuncPtr)(gdImagePtr, int, int); + FuncPtr f; + + if (src->trueColor) { + dst = gdImageCreateTrueColor ( src->sx,src->sy); + f = gdImageGetTrueColorPixel; + } else { + dst = gdImageCreate (src->sx, src->sy); + f = gdImageGetPixel; + } + + if (dst != NULL) { + gdImagePaletteCopy (dst, src); + + for (uY = 0; uY<src->sy; uY++) { + for (uX = 0; uX<src->sx; uX++) { + c = f (src, uX, uY); + gdImageSetPixel(dst, (dst->sx - uX - 1), (dst->sy - uY - 1), c); + } + } + } + + return dst; +} + +/* Rotates an image by 270 degrees (counter clockwise) */ +gdImagePtr gdImageRotate270 ( gdImagePtr src ) +{ + int uY, uX; + int c; + gdImagePtr dst; + typedef int (*FuncPtr)(gdImagePtr, int, int); + FuncPtr f; + + if (src->trueColor) { + dst = gdImageCreateTrueColor (src->sy, src->sx); + f = gdImageGetTrueColorPixel; + } else { + dst = gdImageCreate (src->sy, src->sx); + f = gdImageGetPixel; + } + + if (dst != NULL) { + gdImagePaletteCopy (dst, src); + + for (uY = 0; uY<src->sx; uY++) { + for (uX = 0; uX<src->sx; uX++) { + c = f (src, uX, uY); + gdImageSetPixel(dst, (dst->sx - uY - 1), uX, c); + } + } + } + + return dst; +} + +gdImagePtr gdImageRotate45 (gdImagePtr src, double dAngle, int clrBack) +{ + typedef int (*FuncPtr)(gdImagePtr, int, int); + gdImagePtr dst1,dst2,dst3; + FuncPtr f; + double dRadAngle, dSinE, dTan, dShear; + double dOffset; /* Variable skew offset */ + int u, iShear, newx, newy; + + /* See GEMS I for the algorithm details */ + dRadAngle = dAngle * ROTATE_DEG2RAD; /* Angle in radians */ + dSinE = sin (dRadAngle); + dTan = tan (dRadAngle / 2.0); + + newx = src->sx + src->sy * fabs(dTan); + newy = src->sy; + + /* 1st shear */ + if (src->trueColor) { + dst1 = gdImageCreateTrueColor (newx, newy); + f = gdImageGetTrueColorPixel; + } else { + dst1 = gdImageCreate (newx, newy); + f = gdImageGetPixel; + } + + /******* Perform 1st shear (horizontal) ******/ + if (dst1 == NULL) { + return NULL; + } + + if (dAngle == 0.0) { + /* Returns copy of src */ + gdImageCopy (dst1, src,0,0,0,0,src->sx,src->sy); + return dst1; + } + + gdImagePaletteCopy (dst1, src); + dRadAngle = dAngle * ROTATE_DEG2RAD; /* Angle in radians */ + dSinE = sin (dRadAngle); + dTan = tan (dRadAngle / 2.0); + + for (u = 0; u < dst1->sy; u++) { + if (dTan >= 0.0) { + dShear = ((double)(u + 0.5)) * dTan; + } else { + dShear = ((double)(u - dst1->sy) + 0.5) * dTan; + } + + iShear = floor(dShear); + + gdImageSkewX(dst1, src, u, iShear, (dShear - iShear), clrBack); + } + + /* 2nd shear */ + newx = dst1->sx; + + if (dSinE > 0.0) { + dOffset = (src->sx ) * dSinE; + } else { + dOffset = -dSinE * (src->sx - newx); + } + + newy = (int) ((double) src->sx * fabs( dSinE ) + (double) src->sy * cos (dRadAngle)); + + if (src->trueColor) { + dst2 = gdImageCreateTrueColor (newx, newy); + f = gdImageGetTrueColorPixel; + } else { + dst2 = gdImageCreate (newx, newy); + f = gdImageGetPixel; + } + + if (dst2 == NULL) { + gdImageDestroy(dst1); + return NULL; + } + + for (u = 0; u < dst2->sx; u++, dOffset -= dSinE) { + iShear = floor (dOffset); + gdImageSkewY(dst2, dst1, u, iShear, (dOffset - iShear), clrBack); + } + + + /* 3rd shear */ + gdImageDestroy(dst1); + + newx = (int) ((double)src->sy * fabs (dSinE) + (double)src->sx * cos (dRadAngle)) + 1; + newy = dst2->sy; + + if (src->trueColor) { + dst3 = gdImageCreateTrueColor (newx, newy); + f = gdImageGetTrueColorPixel; + } else { + dst3 = gdImageCreate (newx, newy); + f = gdImageGetPixel; + } + + if (dst3 == NULL) { + gdImageDestroy(dst2); + return NULL; + } + if (dSinE >= 0.0) { + dOffset = (double)(src->sx - 1) * dSinE * -dTan; + } else { + dOffset = dTan * ((double)(src->sx - 1) * -dSinE + (double)(1 - newy)); + } + + for (u = 0; u < dst3->sy; u++, dOffset += dTan) { + int iShear = (int)floor(dOffset); + gdImageSkewX(dst3, dst2, u, iShear, (dOffset - iShear), clrBack); + } + + gdImageDestroy(dst2); + + return dst3; +} + +gdImagePtr gdImageRotate (gdImagePtr src, double dAngle, int clrBack) +{ + gdImagePtr pMidImg; + gdImagePtr rotatedImg; + + if (src == NULL) { + return NULL; + } + + while (dAngle >= 360.0) { + dAngle -= 360.0; + } + + while (dAngle < 0) { + dAngle += 360.0; + } + + if (dAngle == 90.00) { + return gdImageRotate90(src); + } + if (dAngle == 180.00) { + return gdImageRotate180(src); + } + if(dAngle == 270.00) { + return gdImageRotate270 ( src); + } + + if ((dAngle > 45.0) && (dAngle <= 135.0)) { + pMidImg = gdImageRotate90 (src); + dAngle -= 90.0; + } else if ((dAngle > 135.0) && (dAngle <= 225.0)) { + pMidImg = gdImageRotate180 (src); + dAngle -= 180.0; + } else if ((dAngle > 225.0) && (dAngle <= 315.0)) { + pMidImg = gdImageRotate270 (src); + dAngle -= 270.0; + } else { + return gdImageRotate45 (src, dAngle, clrBack); + } + + if (pMidImg == NULL) { + return NULL; + } + + rotatedImg = gdImageRotate45 (pMidImg, dAngle, clrBack); + gdImageDestroy(pMidImg); + + return rotatedImg; +} + gdImagePtr gdImageCreateFromXbm (FILE * fd) { diff --git a/ext/gd/libgd/gd.h b/ext/gd/libgd/gd.h index d0ca82b278..ab38a577c9 100644 --- a/ext/gd/libgd/gd.h +++ b/ext/gd/libgd/gd.h @@ -450,6 +450,12 @@ void gdImageCopyResized(gdImagePtr dst, gdImagePtr src, int dstX, int dstY, int substituted automatically. */ void gdImageCopyResampled(gdImagePtr dst, gdImagePtr src, int dstX, int dstY, int srcX, int srcY, int dstW, int dstH, int srcW, int srcH); +gdImagePtr gdImageRotate90(gdImagePtr src); +gdImagePtr gdImageRotate180(gdImagePtr src); +gdImagePtr gdImageRotate270(gdImagePtr src); +gdImagePtr gdImageRotate45(gdImagePtr src, double dAngle, int clrBack); +gdImagePtr gdImageRotate (gdImagePtr src, double dAngle, int clrBack); + void gdImageSetBrush(gdImagePtr im, gdImagePtr brush); void gdImageSetTile(gdImagePtr im, gdImagePtr tile); void gdImageSetStyle(gdImagePtr im, int *style, int noOfPixels); diff --git a/ext/gd/php_gd.h b/ext/gd/php_gd.h index f70afb43a9..7b7bf341f1 100644 --- a/ext/gd/php_gd.h +++ b/ext/gd/php_gd.h @@ -93,6 +93,11 @@ PHP_FUNCTION(imagecolorresolvealpha); PHP_FUNCTION(imagecolorclosestalpha); PHP_FUNCTION(imagecolorexactalpha); PHP_FUNCTION(imagecopyresampled); + +#ifdef HAVE_GD_BUNDLED +PHP_FUNCTION(imagerotate); +#endif + PHP_FUNCTION(imagesetthickness); PHP_FUNCTION(imagesettile); PHP_FUNCTION(imagecopymergegray); |